home *** CD-ROM | disk | FTP | other *** search
/ Aminet 2 / Aminet AMIGA CDROM (1994)(Walnut Creek)[Feb 1994][W.O. 44790-1].iso / Aminet / util / gnu / groff_src.lha / Groff-1.07 / troff / node.cc < prev    next >
C/C++ Source or Header  |  1992-10-29  |  95KB  |  4,833 lines

  1. // -*- C++ -*-
  2. /* Copyright (C) 1989, 1990, 1991, 1992 Free Software Foundation, Inc.
  3.      Written by James Clark (jjc@jclark.com)
  4.  
  5. This file is part of groff.
  6.  
  7. groff is free software; you can redistribute it and/or modify it under
  8. the terms of the GNU General Public License as published by the Free
  9. Software Foundation; either version 2, or (at your option) any later
  10. version.
  11.  
  12. groff is distributed in the hope that it will be useful, but WITHOUT ANY
  13. WARRANTY; without even the implied warranty of MERCHANTABILITY or
  14. FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
  15. for more details.
  16.  
  17. You should have received a copy of the GNU General Public License along
  18. with groff; see the file COPYING.  If not, write to the Free Software
  19. Foundation, 675 Mass Ave, Cambridge, MA 02139, USA. */
  20.  
  21. #include "troff.h"
  22. #include "symbol.h"
  23. #include "dictionary.h"
  24. #include "hvunits.h"
  25. #include "env.h"
  26. #include "request.h"
  27. #include "node.h"
  28. #include "token.h"
  29. #include "charinfo.h"
  30. #include "font.h"
  31. #include "reg.h"
  32.  
  33. #define STORE_WIDTH 1
  34.  
  35. symbol HYPHEN_SYMBOL("hy");
  36.  
  37. // Character used when a hyphen is inserted at a line break.
  38. static charinfo *soft_hyphen_char;
  39.  
  40. enum constant_space_type { 
  41.   CONSTANT_SPACE_NONE,
  42.   CONSTANT_SPACE_RELATIVE,
  43.   CONSTANT_SPACE_ABSOLUTE
  44.   };
  45.  
  46. struct special_font_list {
  47.   int n;
  48.   special_font_list *next;
  49. };
  50.  
  51. special_font_list *global_special_fonts;
  52. static int global_ligature_mode = 1;
  53. static int global_kern_mode = 1;
  54.  
  55. class track_kerning_function {
  56.   int non_zero;
  57.   units min_size;
  58.   hunits min_amount;
  59.   units max_size;
  60.   hunits max_amount;
  61. public:
  62.   track_kerning_function();
  63.   track_kerning_function(units, hunits, units, hunits);
  64.   int operator==(const track_kerning_function &);
  65.   int operator!=(const track_kerning_function &);
  66.   hunits compute(int point_size);
  67. };
  68.  
  69. // embolden fontno when this is the current font
  70.  
  71. struct conditional_bold {
  72.   conditional_bold *next;
  73.   int fontno;
  74.   hunits offset;
  75.   conditional_bold(int, hunits, conditional_bold * = 0);
  76. };
  77.  
  78. struct tfont;
  79.  
  80. class font_info {
  81.   tfont *last_tfont;
  82.   int number;
  83.   font_size last_size;
  84.   int last_height;
  85.   int last_slant;
  86.   symbol internal_name;
  87.   symbol external_name;
  88.   font *fm;
  89.   char is_bold;
  90.   hunits bold_offset;
  91.   track_kerning_function track_kern;
  92.   constant_space_type is_constant_spaced;
  93.   units constant_space;
  94.   int last_ligature_mode;
  95.   int last_kern_mode;
  96.   conditional_bold *cond_bold_list;
  97.   void flush();
  98. public:
  99.   special_font_list *sf;
  100.   
  101.   font_info(symbol nm, int n, symbol enm, font *f);
  102.   int contains(charinfo *);
  103.   void set_bold(hunits);
  104.   void unbold();
  105.   void set_conditional_bold(int, hunits);
  106.   void conditional_unbold(int);
  107.   void set_track_kern(track_kerning_function &);
  108.   void set_constant_space(constant_space_type, units = 0);
  109.   int is_named(symbol);
  110.   symbol get_name();
  111.   tfont *get_tfont(font_size, int, int, int);
  112.   hunits get_space_width(font_size, int);
  113.   hunits get_narrow_space_width(font_size);
  114.   hunits get_half_narrow_space_width(font_size);
  115.   int get_bold(hunits *);
  116.   int is_special();
  117.   int is_style();
  118. };
  119.  
  120. class tfont_spec {
  121. protected:
  122.   symbol name;
  123.   int input_position;
  124.   font *fm;
  125.   font_size size;
  126.   char is_bold;
  127.   char is_constant_spaced;
  128.   int ligature_mode;
  129.   int kern_mode;
  130.   hunits bold_offset;
  131.   hunits track_kern;            // add this to the width
  132.   hunits constant_space_width;
  133.   int height;
  134.   int slant;
  135. public:
  136.   tfont_spec(symbol nm, int pos, font *, font_size, int, int);
  137.   tfont_spec plain();
  138.   int operator==(const tfont_spec &);
  139.   friend tfont *font_info::get_tfont(font_size fs, int, int, int);
  140. };
  141.  
  142. class tfont : public tfont_spec {
  143.   static tfont *tfont_list;
  144.   tfont *next;
  145.   tfont *plain_version;
  146. public:
  147.   tfont(tfont_spec &);
  148.   int contains(charinfo *);
  149.   hunits get_width(charinfo *c);
  150.   int get_bold(hunits *);
  151.   int get_constant_space(hunits *);
  152.   hunits get_track_kern();
  153.   tfont *get_plain();
  154.   font_size get_size();
  155.   symbol get_name();
  156.   charinfo *get_lig(charinfo *c1, charinfo *c2);
  157.   int get_kern(charinfo *c1, charinfo *c2, hunits *res);
  158.   int get_input_position();
  159.   int get_character_type(charinfo *);
  160.   int get_height();
  161.   int get_slant();
  162.   vunits get_char_height(charinfo *);
  163.   vunits get_char_depth(charinfo *);
  164.   hunits get_char_skew(charinfo *);
  165.   hunits get_italic_correction(charinfo *);
  166.   hunits get_left_italic_correction(charinfo *);
  167.   hunits get_subscript_correction(charinfo *);
  168.   friend tfont *make_tfont(tfont_spec &);
  169. };
  170.  
  171. inline int env_definite_font(environment *env)
  172. {
  173.   return env->get_family()->make_definite(env->get_font());
  174. }
  175.  
  176. static void invalidate_fontno(int n);
  177.  
  178. /* font_info functions */
  179.  
  180. static font_info **font_table = 0;
  181. static int font_table_size = 0;
  182.  
  183. font_info::font_info(symbol nm, int n, symbol enm, font *f)
  184. : internal_name(nm), external_name(enm), fm(f), number(n),
  185.   is_constant_spaced(CONSTANT_SPACE_NONE),
  186.   sf(0), is_bold(0), cond_bold_list(0),
  187.   last_ligature_mode(1), last_kern_mode(1),
  188.   last_tfont(0), last_size(0)
  189. {
  190. }
  191.  
  192. inline int font_info::contains(charinfo *ci)
  193. {
  194.   return fm != 0 && fm->contains(ci->get_index());
  195. }
  196.  
  197. inline int font_info::is_special()
  198. {
  199.   return fm != 0 && fm->is_special();
  200. }
  201.  
  202. inline int font_info::is_style()
  203. {
  204.   return fm == 0;
  205. }
  206.  
  207. // this is the current_font, fontno is where we found the character,
  208. // presumably a special font
  209.  
  210. tfont *font_info::get_tfont(font_size fs, int height, int slant, int fontno)
  211. {
  212.   if (last_tfont == 0 || fs != last_size
  213.       || height != last_height || slant != last_slant
  214.       || global_ligature_mode != last_ligature_mode
  215.       || global_kern_mode != last_kern_mode
  216.       || fontno != number) {
  217.     font_info *f = font_table[fontno];
  218.     tfont_spec spec(f->external_name, f->number, f->fm, fs, height, slant);
  219.     for (conditional_bold *p = cond_bold_list; p; p = p->next)
  220.       if (p->fontno == fontno) {
  221.         spec.is_bold = 1;
  222.         spec.bold_offset = p->offset;
  223.         break;
  224.       }
  225.     if (!spec.is_bold && is_bold) {
  226.       spec.is_bold = 1;
  227.       spec.bold_offset = bold_offset;
  228.     }
  229.     spec.track_kern = track_kern.compute(fs.to_scaled_points());
  230.     spec.ligature_mode = global_ligature_mode;
  231.     spec.kern_mode = global_kern_mode;
  232.     switch (is_constant_spaced) {
  233.     case CONSTANT_SPACE_NONE:
  234.       break;
  235.     case CONSTANT_SPACE_ABSOLUTE:
  236.       spec.is_constant_spaced = 1;
  237.       spec.constant_space_width = constant_space;
  238.       break;
  239.     case CONSTANT_SPACE_RELATIVE:
  240.       spec.is_constant_spaced = 1;
  241.       spec.constant_space_width
  242.         = scale(constant_space*fs.to_scaled_points(),
  243.             units_per_inch,
  244.             36*72*sizescale);
  245.       break;
  246.     default:
  247.       assert(0);
  248.     }
  249.     if (fontno != number)
  250.       return make_tfont(spec);
  251.     last_tfont = make_tfont(spec);
  252.     last_size = fs;
  253.     last_height = height;
  254.     last_slant = slant;
  255.     last_ligature_mode = global_ligature_mode;
  256.     last_kern_mode = global_kern_mode;
  257.       }
  258.   return last_tfont;
  259. }
  260.  
  261. int font_info::get_bold(hunits *res)
  262. {
  263.   if (is_bold) {
  264.     *res = bold_offset;
  265.     return 1;
  266.   }
  267.   else
  268.     return 0;
  269. }
  270.  
  271. void font_info::unbold()
  272. {
  273.   if (is_bold) {
  274.     is_bold = 0;
  275.     flush();
  276.   }
  277. }
  278.  
  279. void font_info::set_bold(hunits offset)
  280. {
  281.   if (!is_bold || offset != bold_offset) {
  282.     is_bold = 1;
  283.     bold_offset = offset;
  284.     flush();
  285.   }
  286. }
  287.  
  288. void font_info::set_conditional_bold(int fontno, hunits offset)
  289. {
  290.   for (conditional_bold *p = cond_bold_list; p; p = p->next)
  291.     if (p->fontno == fontno) {
  292.       if (offset != p->offset) {
  293.     p->offset = offset;
  294.     flush();
  295.       }
  296.       return;
  297.     }
  298.   cond_bold_list = new conditional_bold(fontno, offset, cond_bold_list);
  299. }
  300.  
  301. conditional_bold::conditional_bold(int f, hunits h, conditional_bold *x)
  302.      : fontno(f), offset(h), next(x)
  303. {
  304. }
  305.  
  306. void font_info::conditional_unbold(int fontno)
  307. {
  308.   for (conditional_bold **p = &cond_bold_list; *p; p = &(*p)->next)
  309.     if ((*p)->fontno == fontno) {
  310.       conditional_bold *tem = *p;
  311.       *p = (*p)->next;
  312.       delete tem;
  313.       flush();
  314.       return;
  315.     }
  316. }
  317.                  
  318. void font_info::set_constant_space(constant_space_type type, units x)
  319. {
  320.   if (type != is_constant_spaced
  321.       || (type != CONSTANT_SPACE_NONE && x != constant_space)) {
  322.     flush();
  323.     is_constant_spaced = type;
  324.     constant_space = x;
  325.       }
  326. }
  327.  
  328. void font_info::set_track_kern(track_kerning_function  &tk)
  329. {
  330.   if (track_kern != tk) {
  331.     track_kern = tk;
  332.     flush();
  333.   }
  334. }
  335.  
  336. void font_info::flush()
  337. {
  338.   last_tfont = 0;
  339. }
  340.  
  341. int font_info::is_named(symbol s)
  342. {
  343.   return internal_name == s;
  344. }
  345.  
  346. symbol font_info::get_name()
  347. {
  348.   return internal_name;
  349. }
  350.  
  351. hunits font_info::get_space_width(font_size fs, int space_size)
  352. {
  353.   if (is_constant_spaced == CONSTANT_SPACE_NONE)
  354.     return scale(hunits(fm->get_space_width(fs.to_scaled_points())),
  355.             space_size, 12);
  356.   else if (is_constant_spaced == CONSTANT_SPACE_ABSOLUTE)
  357.     return constant_space;
  358.   else
  359.     return scale(constant_space*fs.to_scaled_points(),
  360.          units_per_inch, 36*72*sizescale);
  361. }
  362.  
  363. hunits font_info::get_narrow_space_width(font_size fs)
  364. {
  365.   charinfo *ci = get_charinfo(symbol("|"));
  366.   if (fm->contains(ci->get_index()))
  367.     return hunits(fm->get_width(ci->get_index(), fs.to_scaled_points()));
  368.   else
  369.     return hunits(fs.to_units()/6);
  370. }
  371.  
  372. hunits font_info::get_half_narrow_space_width(font_size fs)
  373. {
  374.   charinfo *ci = get_charinfo(symbol("^"));
  375.   if (fm->contains(ci->get_index()))
  376.     return hunits(fm->get_width(ci->get_index(), fs.to_scaled_points()));
  377.   else
  378.     return hunits(fs.to_units()/12);
  379. }
  380.  
  381. /* tfont */
  382.  
  383. tfont_spec::tfont_spec(symbol nm, int n, font *f, 
  384.                font_size s, int h, int sl)
  385.      : name(nm), input_position(n), fm(f), size(s),
  386.      is_bold(0), is_constant_spaced(0), ligature_mode(1), kern_mode(1),
  387.      height(h), slant(sl)
  388. {
  389.   if (height == size.to_scaled_points())
  390.     height = 0;
  391. }
  392.  
  393. int tfont_spec::operator==(const tfont_spec &spec)
  394. {
  395.   if (fm == spec.fm 
  396.       && size == spec.size
  397.       && input_position == spec.input_position
  398.       && name == spec.name
  399.       && height == spec.height
  400.       && slant == spec.slant
  401.       && (is_bold 
  402.       ? (spec.is_bold && bold_offset == spec.bold_offset)
  403.       : !spec.is_bold)
  404.       && track_kern == spec.track_kern
  405.       && (is_constant_spaced
  406.       ? (spec.is_constant_spaced 
  407.          && constant_space_width == spec.constant_space_width)
  408.       : !spec.is_constant_spaced)
  409.       && ligature_mode == spec.ligature_mode
  410.       && kern_mode == spec.kern_mode)
  411.     return 1;
  412.   else
  413.     return 0;
  414. }
  415.  
  416. tfont_spec tfont_spec::plain()
  417. {
  418.   return tfont_spec(name, input_position, fm, size, height, slant);
  419. }
  420.  
  421. hunits tfont::get_width(charinfo *c)
  422. {
  423.   if (is_constant_spaced)
  424.     return constant_space_width;
  425.   else if (is_bold)
  426.     return (hunits(fm->get_width(c->get_index(), size.to_scaled_points()))
  427.         + track_kern + bold_offset);
  428.   else
  429.     return (hunits(fm->get_width(c->get_index(), size.to_scaled_points())) + track_kern);
  430. }
  431.  
  432. vunits tfont::get_char_height(charinfo *c)
  433. {
  434.   vunits v = fm->get_height(c->get_index(), size.to_scaled_points());
  435.   if (height != 0 && height != size.to_scaled_points())
  436.     return scale(v, height, size.to_scaled_points());
  437.   else
  438.     return v;
  439. }
  440.  
  441. vunits tfont::get_char_depth(charinfo *c)
  442. {
  443.   vunits v = fm->get_depth(c->get_index(), size.to_scaled_points());
  444.   if (height != 0 && height != size.to_scaled_points())
  445.     return scale(v, height, size.to_scaled_points());
  446.   else
  447.     return v;
  448. }
  449.  
  450. hunits tfont::get_char_skew(charinfo *c)
  451. {
  452.   return hunits(fm->get_skew(c->get_index(), size.to_scaled_points(), slant));
  453. }
  454.  
  455. hunits tfont::get_italic_correction(charinfo *c)
  456. {
  457.   return hunits(fm->get_italic_correction(c->get_index(), size.to_scaled_points()));
  458. }
  459.  
  460. hunits tfont::get_left_italic_correction(charinfo *c)
  461. {
  462.   return hunits(fm->get_left_italic_correction(c->get_index(),
  463.                            size.to_scaled_points()));
  464. }
  465.  
  466. hunits tfont::get_subscript_correction(charinfo *c)
  467. {
  468.   return hunits(fm->get_subscript_correction(c->get_index(),
  469.                          size.to_scaled_points()));
  470. }
  471.  
  472. inline int tfont::get_input_position()
  473. {
  474.   return input_position;
  475. }
  476.  
  477. inline int tfont::contains(charinfo *ci)
  478. {
  479.   return fm->contains(ci->get_index());
  480. }
  481.  
  482. inline int tfont::get_character_type(charinfo *ci)
  483. {
  484.   return fm->get_character_type(ci->get_index());
  485. }
  486.  
  487. inline int tfont::get_bold(hunits *res)
  488. {
  489.   if (is_bold) {
  490.     *res = bold_offset;
  491.     return 1;
  492.   }
  493.   else
  494.     return 0;
  495. }
  496.  
  497. inline int tfont::get_constant_space(hunits *res)
  498. {
  499.   if (is_constant_spaced) {
  500.     *res = constant_space_width;
  501.     return 1;
  502.   }
  503.   else
  504.     return 0;
  505. }
  506.  
  507. inline hunits tfont::get_track_kern()
  508. {
  509.   return track_kern;
  510. }
  511.  
  512. inline tfont *tfont::get_plain()
  513. {
  514.   return plain_version;
  515. }
  516.  
  517. inline font_size tfont::get_size()
  518. {
  519.   return size;
  520. }
  521.  
  522. inline symbol tfont::get_name()
  523. {
  524.   return name;
  525. }
  526.  
  527. inline int tfont::get_height()
  528. {
  529.   return height;
  530. }
  531.  
  532. inline int tfont::get_slant()
  533. {
  534.   return slant;
  535. }
  536.  
  537. symbol SYMBOL_ff("ff");
  538. symbol SYMBOL_fi("fi");
  539. symbol SYMBOL_fl("fl");
  540. symbol SYMBOL_Fi("Fi");
  541. symbol SYMBOL_Fl("Fl");
  542.  
  543. charinfo *tfont::get_lig(charinfo *c1, charinfo *c2)
  544. {
  545.   if (ligature_mode == 0)
  546.     return 0;
  547.   charinfo *ci = 0;
  548.   if (c1->get_ascii_code() == 'f') {
  549.     switch (c2->get_ascii_code()) {
  550.     case 'f':
  551.       if (fm->has_ligature(font::LIG_ff))
  552.     ci = get_charinfo(SYMBOL_ff);
  553.       break;
  554.     case 'i':
  555.       if (fm->has_ligature(font::LIG_fi))
  556.     ci = get_charinfo(SYMBOL_fi);
  557.       break;
  558.     case 'l':
  559.       if (fm->has_ligature(font::LIG_fl))
  560.     ci = get_charinfo(SYMBOL_fl);
  561.       break;
  562.     }
  563.   }
  564.   else if (ligature_mode != 2 && c1->nm == SYMBOL_ff) {
  565.     switch (c2->get_ascii_code()) {
  566.     case 'i':
  567.       if (fm->has_ligature(font::LIG_ffi))
  568.     ci = get_charinfo(SYMBOL_Fi);
  569.       break;
  570.     case 'l':
  571.       if (fm->has_ligature(font::LIG_ffl))
  572.     ci = get_charinfo(SYMBOL_Fl);
  573.       break;
  574.     }
  575.   }
  576.   if (ci != 0 && fm->contains(ci->get_index()))
  577.     return ci;
  578.   return 0;
  579. }
  580.  
  581. inline int tfont::get_kern(charinfo *c1, charinfo *c2, hunits *res)
  582. {
  583.   if (kern_mode == 0)
  584.     return 0;
  585.   else {
  586.     int n = fm->get_kern(c1->get_index(),
  587.              c2->get_index(),
  588.              size.to_scaled_points());
  589.     if (n) {
  590.       *res = hunits(n);
  591.       return 1;
  592.     }
  593.     else
  594.       return 0;
  595.   }
  596. }
  597.  
  598. tfont *make_tfont(tfont_spec &spec)
  599. {
  600.   for (tfont *p = tfont::tfont_list; p; p = p->next)
  601.     if (*p == spec)
  602.       return p;
  603.   return new tfont(spec);
  604. }
  605.  
  606. tfont *tfont::tfont_list = 0;
  607.  
  608. tfont::tfont(tfont_spec &spec) : tfont_spec(spec)
  609. {
  610.   next = tfont_list;
  611.   tfont_list = this;
  612.   tfont_spec plain_spec = plain();
  613.   for (tfont *p = tfont_list; p; p = p->next)
  614.     if (*p == plain_spec) {
  615.       plain_version = p;
  616.       break;
  617.     }
  618.   if (!p)
  619.     plain_version = new tfont(plain_spec);
  620. }
  621.  
  622. /* output_file */
  623.  
  624. class real_output_file : public output_file {
  625.   int piped;
  626.   int printing;
  627.   virtual void really_transparent_char(unsigned char) = 0;
  628.   virtual void really_print_line(hunits x, vunits y, node *n,
  629.                  vunits before, vunits after) = 0;
  630.   virtual void really_begin_page(int pageno, vunits page_length) = 0;
  631.   virtual void really_copy_file(hunits x, vunits y, const char *filename);
  632. protected:
  633.   FILE *fp;
  634. public:
  635.   real_output_file();
  636.   ~real_output_file();
  637.   void flush();
  638.   void transparent_char(unsigned char);
  639.   void print_line(hunits x, vunits y, node *n, vunits before, vunits after);
  640.   void begin_page(int pageno, vunits page_length);
  641.   int is_printing();
  642.   void copy_file(hunits x, vunits y, const char *filename);
  643. };
  644.  
  645. class suppress_output_file : public real_output_file {
  646. public:
  647.   suppress_output_file();
  648.   void really_transparent_char(unsigned char);
  649.   void really_print_line(hunits x, vunits y, node *n, vunits, vunits);
  650.   void really_begin_page(int pageno, vunits page_length);
  651. };
  652.  
  653. class ascii_output_file : public real_output_file {
  654. public:
  655.   ascii_output_file();
  656.   void really_transparent_char(unsigned char);
  657.   void really_print_line(hunits x, vunits y, node *n, vunits, vunits);
  658.   void really_begin_page(int pageno, vunits page_length);
  659.   void outc(unsigned char c);
  660.   void outs(const char *s);
  661. };
  662.  
  663. void ascii_output_file::outc(unsigned char c)
  664. {
  665.   fputc(c, fp);
  666. }
  667.  
  668. void ascii_output_file::outs(const char *s)
  669. {
  670.   fputc('<', fp);
  671.   if (s)
  672.     fputs(s, fp);
  673.   fputc('>', fp);
  674. }
  675.  
  676. struct hvpair;
  677.       
  678. class troff_output_file : public real_output_file {
  679.   units hpos;
  680.   units vpos;
  681.   units output_vpos;
  682.   units output_hpos;
  683.   int force_motion;
  684.   int current_size;
  685.   int current_slant;
  686.   int current_height;
  687.   tfont *current_tfont;
  688.   int current_font_number;
  689.   symbol *font_position;
  690.   int nfont_positions;
  691.   enum { TBUF_SIZE = 256 };
  692.   char tbuf[TBUF_SIZE];
  693.   int tbuf_len;
  694.   int tbuf_kern;
  695.   void do_motion();
  696.   vunits page_length;
  697.   void put(char c);
  698.   void put(unsigned char c);
  699.   void put(int i);
  700.   void put(const char *s);
  701.   void set_font(tfont *tf);
  702.   void flush_tbuf();
  703. public:
  704.   troff_output_file();
  705.   ~troff_output_file();
  706.   void put_char(charinfo *ci, tfont *tf);
  707.   void put_char_width(charinfo *ci, tfont *tf, hunits w, hunits k);
  708.   void right(hunits);
  709.   void down(vunits);
  710.   void moveto(hunits, vunits);
  711.   void start_special();
  712.   void special_char(unsigned char c);
  713.   void end_special();
  714.   void word_marker();
  715.   void really_transparent_char(unsigned char c);
  716.   void really_print_line(hunits x, vunits y, node *n, vunits before, vunits after);
  717.   void really_begin_page(int pageno, vunits page_length);
  718.   void really_copy_file(hunits x, vunits y, const char *filename);
  719.   void draw(char, hvpair *, int, font_size);
  720.   int get_hpos() { return hpos; }
  721.   int get_vpos() { return vpos; }
  722. };
  723.  
  724. static void put_string(const char *s, FILE *fp)
  725. {
  726.   for (; *s != '\0'; ++s)
  727.     putc(*s, fp);
  728. }
  729.  
  730. inline void troff_output_file::put(char c)
  731. {
  732.   putc(c, fp);
  733. }
  734.  
  735. inline void troff_output_file::put(unsigned char c)
  736. {
  737.   putc(c, fp);
  738. }
  739.  
  740. inline void troff_output_file::put(const char *s)
  741. {
  742.   put_string(s, fp);
  743. }
  744.  
  745. inline void troff_output_file::put(int i)
  746. {
  747.   put_string(itoa(i), fp);
  748. }
  749.  
  750. void troff_output_file::start_special()
  751. {
  752.   flush_tbuf();
  753.   do_motion();
  754.   put("x X ");
  755. }
  756.  
  757. void troff_output_file::special_char(unsigned char c)
  758. {
  759.   put(c);
  760.   if (c == '\n')
  761.     put('+');
  762. }
  763.  
  764. void troff_output_file::end_special()
  765. {
  766.   put('\n');
  767. }
  768.  
  769. inline void troff_output_file::moveto(hunits h, vunits v)
  770. {
  771.   hpos = h.to_units();
  772.   vpos = v.to_units();
  773. }
  774.  
  775. void troff_output_file::really_print_line(hunits x, vunits y, node *n,
  776.                       vunits before, vunits after)
  777. {
  778.   moveto(x, y);
  779.   while (n != 0) {
  780.     n->tprint(this);
  781.     n = n->next;
  782.   }
  783.   flush_tbuf();
  784.   // This ensures that transparent throughput will have a more predictable
  785.   // position.
  786.   do_motion();
  787.   force_motion = 1;
  788.   hpos = 0;
  789.   put('n');
  790.   put(before.to_units());
  791.   put(' ');
  792.   put(after.to_units());
  793.   put('\n');
  794. }
  795.  
  796. inline void troff_output_file::word_marker()
  797. {
  798.   flush_tbuf();
  799.   put('w');
  800. }
  801.  
  802. inline void troff_output_file::right(hunits n)
  803. {
  804.   hpos += n.to_units();
  805. }
  806.  
  807. inline void troff_output_file::down(vunits n)
  808. {
  809.   vpos += n.to_units();
  810. }
  811.  
  812. void troff_output_file::do_motion()
  813. {
  814.   if (force_motion) {
  815.     put('V');
  816.     put(vpos);
  817.     put('\n');
  818.     put('H');
  819.     put(hpos);
  820.     put('\n');
  821.   }
  822.   else {
  823.     if (hpos != output_hpos) {
  824.       units n = hpos - output_hpos;
  825.       if (n > 0 && n < hpos) {
  826.     put('h');
  827.     put(n);
  828.       }
  829.       else {
  830.     put('H');
  831.     put(hpos);
  832.       }
  833.       put('\n');
  834.     }
  835.     if (vpos != output_vpos) {
  836.       units n = vpos - output_vpos;
  837.       if (n > 0 && n < vpos) {
  838.     put('v');
  839.     put(n);
  840.       }
  841.       else {
  842.     put('V');
  843.     put(vpos);
  844.       }
  845.       put('\n');
  846.     }
  847.   }
  848.   output_vpos = vpos;
  849.   output_hpos = hpos;
  850.   force_motion = 0;
  851. }
  852.  
  853. void troff_output_file::flush_tbuf()
  854. {
  855.   if (tbuf_len == 0)
  856.     return;
  857.   if (tbuf_kern == 0)
  858.     put('t');
  859.   else {
  860.     put('u');
  861.     put(tbuf_kern);
  862.     put(' ');
  863.   }
  864.   for (int i = 0; i < tbuf_len; i++)
  865.     put(tbuf[i]);
  866.   put('\n');
  867.   tbuf_len = 0;
  868. }
  869.  
  870. void troff_output_file::put_char_width(charinfo *ci, tfont *tf, hunits w,
  871.                        hunits k)
  872. {
  873.   if (tf != current_tfont) {
  874.     flush_tbuf();
  875.     set_font(tf);
  876.   }
  877.   char c = ci->get_ascii_code();
  878.   int kk = k.to_units();
  879.   if (c == '\0') {
  880.     flush_tbuf();
  881.     do_motion();
  882.     if (ci->numbered()) {
  883.       put('N');
  884.       put(ci->get_number());
  885.     }
  886.     else {
  887.       put('C');
  888.       const char *s = ci->nm.contents();
  889.       if (s[1] == 0) {
  890.     put('\\');
  891.     put(s[0]);
  892.       }
  893.       else
  894.     put(s);
  895.     }
  896.     put('\n');
  897.     hpos += w.to_units() + kk;
  898.   }
  899.   else if (tcommand_flag) {
  900.     if (tbuf_len > 0 && hpos == output_hpos && vpos == output_vpos
  901.     && kk == tbuf_kern
  902.     && tbuf_len < TBUF_SIZE) {
  903.       tbuf[tbuf_len++] = c;
  904.       output_hpos += w.to_units() + kk;
  905.       hpos = output_hpos;
  906.       return;
  907.     }
  908.     flush_tbuf();
  909.     do_motion();
  910.     tbuf[tbuf_len++] = c;
  911.     output_hpos += w.to_units() + kk;
  912.     tbuf_kern = kk;
  913.     hpos = output_hpos;
  914.   }
  915.   else {
  916.     // flush_tbuf();
  917.     int n = hpos - output_hpos;
  918.     if (vpos == output_vpos && n > 0 && n < 100 && !force_motion) {
  919.       put(char(n/10 + '0'));
  920.       put(char(n%10 + '0'));
  921.       put(c);
  922.       output_hpos = hpos;
  923.     }
  924.     else {
  925.       do_motion();
  926.       put('c');
  927.       put(c);
  928.     }
  929.     hpos += w.to_units() + kk;
  930.   }
  931. }
  932.  
  933. void troff_output_file::put_char(charinfo *ci, tfont *tf)
  934. {
  935.   flush_tbuf();
  936.   if (tf != current_tfont)
  937.     set_font(tf);
  938.   char c = ci->get_ascii_code();
  939.   if (c == '\0') {
  940.     do_motion();
  941.     if (ci->numbered()) {
  942.       put('N');
  943.       put(ci->get_number());
  944.     }
  945.     else {
  946.       put('C');
  947.       const char *s = ci->nm.contents();
  948.       if (s[1] == 0) {
  949.     put('\\');
  950.     put(s[0]);
  951.       }
  952.       else
  953.     put(s);
  954.     }
  955.     put('\n');
  956.   }
  957.   else {
  958.     int n = hpos - output_hpos;
  959.     if (vpos == output_vpos && n > 0 && n < 100) {
  960.       put(char(n/10 + '0'));
  961.       put(char(n%10 + '0'));
  962.       put(c);
  963.       output_hpos = hpos;
  964.     }
  965.     else {
  966.       do_motion();
  967.       put('c');
  968.       put(c);
  969.     }
  970.   }
  971. }
  972.  
  973. void troff_output_file::set_font(tfont *tf)
  974. {
  975.   if (current_tfont == tf)
  976.     return;
  977.   int n = tf->get_input_position();
  978.   symbol nm = tf->get_name();
  979.   if (n >= nfont_positions || font_position[n] != nm) {
  980.     put("x font ");
  981.     put(n);
  982.     put(' ');
  983.     put(nm.contents());
  984.     put('\n');
  985.     if (n >= nfont_positions) {
  986.       int old_nfont_positions = nfont_positions;
  987.       symbol *old_font_position = font_position;
  988.       nfont_positions *= 3;
  989.       nfont_positions /= 2;
  990.       if (nfont_positions <= n)
  991.     nfont_positions = n + 10;
  992.       font_position = new symbol[nfont_positions];
  993.       memcpy(font_position, old_font_position,
  994.          old_nfont_positions*sizeof(symbol));
  995.       a_delete old_font_position;
  996.     }
  997.     font_position[n] = nm;
  998.   }
  999.   if (current_font_number != n) {
  1000.     put('f');
  1001.     put(n);
  1002.     put('\n');
  1003.     current_font_number = n;
  1004.   }
  1005.   int size = tf->get_size().to_scaled_points();
  1006.   if (current_size != size) {
  1007.     put('s');
  1008.     put(size);
  1009.     put('\n');
  1010.     current_size = size;
  1011.   }
  1012.   int slant = tf->get_slant();
  1013.   if (current_slant != slant) {
  1014.     put("x Slant ");
  1015.     put(slant);
  1016.     put('\n');
  1017.     current_slant = slant;
  1018.   }
  1019.   int height = tf->get_height();
  1020.   if (current_height != height) {
  1021.     put("x Height ");
  1022.     put(height == 0 ? current_size : height);
  1023.     put('\n');
  1024.     current_height = height;
  1025.   }
  1026.   current_tfont = tf;
  1027. }
  1028.  
  1029. void troff_output_file::draw(char code, hvpair *point, int npoints,
  1030.                  font_size fsize)
  1031. {
  1032.   flush_tbuf();
  1033.   do_motion();
  1034.   int size = fsize.to_scaled_points();
  1035.   if (current_size != size) {
  1036.     put('s');
  1037.     put(size);
  1038.     put('\n');
  1039.     current_size = size;
  1040.     current_tfont = 0;
  1041.   }
  1042.   put('D');
  1043.   put(code);
  1044.   int i;
  1045.   if (code == 'c') {
  1046.     put(' ');
  1047.     put(point[0].h.to_units());
  1048.   }
  1049.   else
  1050.     for (i = 0; i < npoints; i++) {
  1051.       put(' ');
  1052.       put(point[i].h.to_units());
  1053.       put(' ');
  1054.       put(point[i].v.to_units());
  1055.     }
  1056.   for (i = 0; i < npoints; i++)
  1057.     output_hpos += point[i].h.to_units();
  1058.   hpos = output_hpos;
  1059.   if (code != 'e') {
  1060.     for (i = 0; i < npoints; i++)
  1061.       output_vpos += point[i].v.to_units();
  1062.     vpos = output_vpos;
  1063.   }
  1064.   put('\n');
  1065. }
  1066.  
  1067. void troff_output_file::really_begin_page(int pageno, vunits pl)
  1068. {
  1069.   flush_tbuf();
  1070.   if (page_length > V0) {
  1071.     put('V');
  1072.     put(page_length.to_units());
  1073.     put('\n');
  1074.   }
  1075.   page_length = pl;
  1076.   current_tfont = 0;
  1077.   current_font_number = -1;
  1078.   current_size = 0;
  1079.   // current_height = 0;
  1080.   // current_slant = 0;
  1081.   hpos = 0;
  1082.   vpos = 0;
  1083.   output_hpos = 0;
  1084.   output_vpos = 0;
  1085.   force_motion = 1;
  1086.   for (int i = 0; i < nfont_positions; i++)
  1087.     font_position[i] = NULL_SYMBOL;
  1088.   put('p');
  1089.   put(pageno);
  1090.   put('\n');
  1091. }
  1092.  
  1093. void troff_output_file::really_copy_file(hunits x, vunits y, const char *filename)
  1094. {
  1095.   moveto(x, y);
  1096.   flush_tbuf();
  1097.   do_motion();
  1098.   errno = 0;
  1099.   FILE *ifp = fopen(filename, "r");
  1100.   if (ifp == 0)
  1101.     error("can't open `%1': %2", filename, strerror(errno));
  1102.   else {
  1103.     int c;
  1104.     while ((c = getc(ifp)) != EOF)
  1105.       put(char(c));
  1106.     fclose(ifp);
  1107.   }
  1108.   force_motion = 1;
  1109.   current_size = 0;
  1110.   current_tfont = 0;
  1111.   current_font_number = -1;
  1112.   for (int i = 0; i < nfont_positions; i++)
  1113.     font_position[i] = NULL_SYMBOL;
  1114. }
  1115.   
  1116. void troff_output_file::really_transparent_char(unsigned char c)
  1117. {
  1118.   put(c);
  1119. }
  1120.  
  1121. troff_output_file::~troff_output_file()
  1122. {
  1123.   flush_tbuf();
  1124.   if (page_length > V0) {
  1125.     put("x trailer\n");
  1126.     put('V');
  1127.     put(page_length.to_units());
  1128.     put('\n');
  1129.   }
  1130.   put("x stop\n");
  1131.   a_delete font_position;
  1132. }
  1133.  
  1134. troff_output_file::troff_output_file()
  1135. : current_height(0), current_slant(0), tbuf_len(0), nfont_positions(10)
  1136. {
  1137.   font_position = new symbol[nfont_positions];
  1138.   put("x T ");
  1139.   put(device);
  1140.   put('\n');
  1141.   put("x res ");
  1142.   put(units_per_inch);
  1143.   put(' ');
  1144.   put(hresolution);
  1145.   put(' ');
  1146.   put(vresolution);
  1147.   put('\n');
  1148.   put("x init\n");
  1149. }
  1150.  
  1151. /* output_file */
  1152.  
  1153. output_file *the_output = 0;
  1154.  
  1155. output_file::output_file()
  1156. {
  1157. }
  1158.  
  1159. output_file::~output_file()
  1160. {
  1161. }
  1162.  
  1163. real_output_file::real_output_file()
  1164. : printing(0)
  1165. {
  1166.   if (pipe_command) {
  1167.     if ((fp = popen(pipe_command, "w")) != 0) {
  1168.       piped = 1;
  1169.       return;
  1170.     }
  1171.     error("pipe open failed: %1", strerror(errno));
  1172.   }
  1173.   piped = 0;
  1174.   fp = stdout;
  1175. }
  1176.  
  1177. real_output_file::~real_output_file()
  1178. {
  1179.   if (!fp)
  1180.     return;
  1181.   // To avoid looping, set fp to 0 before calling fatal().
  1182.   if (ferror(fp) || fflush(fp) < 0) {
  1183.     fp = 0;
  1184.     fatal("error writing output file");
  1185.   }
  1186.   if (piped) {
  1187.     int result = pclose(fp);
  1188.     fp = 0;
  1189.     if (result < 0)
  1190.       fatal("pclose failed");
  1191.     if ((result & 0x7f) != 0)
  1192.       error("output process `%1' got fatal signal %2",
  1193.         pipe_command, result & 0x7f);
  1194.     else {
  1195.       int exit_status = (result >> 8) & 0xff;
  1196.       if (exit_status != 0)
  1197.     error("output process `%1' exited with status %2",
  1198.           pipe_command, exit_status);
  1199.     }
  1200.   }
  1201.   else if (fclose(fp) < 0) {
  1202.     fp = 0;
  1203.     fatal("error closing output file");
  1204.   }
  1205. }
  1206.  
  1207. void real_output_file::flush()
  1208. {
  1209.   if (fflush(fp) < 0)
  1210.     fatal("error writing output file");
  1211. }
  1212.  
  1213. int real_output_file::is_printing()
  1214. {
  1215.   return printing;
  1216. }
  1217.  
  1218. void real_output_file::begin_page(int pageno, vunits page_length)
  1219. {
  1220.   printing = in_output_page_list(pageno);
  1221.   if (printing)
  1222.     really_begin_page(pageno, page_length);
  1223. }
  1224.  
  1225. void real_output_file::copy_file(hunits x, vunits y, const char *filename)
  1226. {
  1227.   if (printing)
  1228.     really_copy_file(x, y, filename);
  1229. }
  1230.  
  1231. void real_output_file::transparent_char(unsigned char c)
  1232. {
  1233.   if (printing)
  1234.     really_transparent_char(c);
  1235. }
  1236.  
  1237. void real_output_file::print_line(hunits x, vunits y, node *n,
  1238.                  vunits before, vunits after)
  1239. {
  1240.   if (printing)
  1241.     really_print_line(x, y, n, before, after);
  1242.   delete_node_list(n);
  1243. }
  1244.  
  1245. void real_output_file::really_copy_file(hunits, vunits, const char *)
  1246. {
  1247.   // do nothing
  1248. }
  1249.  
  1250.  
  1251. /* ascii_output_file */
  1252.  
  1253. void ascii_output_file::really_transparent_char(unsigned char c)
  1254. {
  1255.   putc(c, fp);
  1256. }
  1257.  
  1258. void ascii_output_file::really_print_line(hunits, vunits, node *n, vunits, vunits)
  1259. {
  1260.   while (n != 0) {
  1261.     n->ascii_print(this);
  1262.     n = n->next;
  1263.   }
  1264.   fputc('\n', fp);
  1265. }
  1266.  
  1267. void ascii_output_file::really_begin_page(int /*pageno*/, vunits /*page_length*/)
  1268. {
  1269.   fputs("<beginning of page>\n", fp);
  1270. }
  1271.  
  1272. ascii_output_file::ascii_output_file()
  1273. {
  1274. }
  1275.  
  1276. /* suppress_output_file */
  1277.  
  1278. suppress_output_file::suppress_output_file()
  1279. {
  1280. }
  1281.  
  1282. void suppress_output_file::really_print_line(hunits, vunits, node *, vunits, vunits)
  1283. {
  1284. }
  1285.  
  1286. void suppress_output_file::really_begin_page(int, vunits)
  1287. {
  1288. }
  1289.  
  1290. void suppress_output_file::really_transparent_char(unsigned char)
  1291. {
  1292. }
  1293.  
  1294. /* glyphs, ligatures, kerns, discretionary breaks */
  1295.  
  1296. class glyph_node : public node {
  1297.   static glyph_node *free_list;
  1298. protected:
  1299.   charinfo *ci;
  1300.   tfont *tf;
  1301. #ifdef STORE_WIDTH
  1302.   hunits wid;
  1303.   glyph_node(charinfo *, tfont *, hunits, node * = 0);
  1304. #endif
  1305. public:
  1306.   void *operator new(size_t);
  1307.   void operator delete(void *);
  1308.   glyph_node(charinfo *, tfont *, node * = 0);
  1309.   ~glyph_node() {}
  1310.   node *copy();
  1311.   node *merge_glyph_node(glyph_node *);
  1312.   node *merge_self(node *);
  1313.   hunits width();
  1314.   node *last_char_node();
  1315.   units size();
  1316.   void vertical_extent(vunits *, vunits *);
  1317.   hunits subscript_correction();
  1318.   hunits italic_correction();
  1319.   hunits left_italic_correction();
  1320.   hunits skew();
  1321.   hyphenation_type get_hyphenation_type();
  1322.   tfont *get_tfont();
  1323.   void tprint(troff_output_file *);
  1324.   void zero_width_tprint(troff_output_file *);
  1325.   hyphen_list *get_hyphen_list(hyphen_list *ss = 0);
  1326.   node *add_self(node *, hyphen_list **);
  1327.   int ends_sentence();
  1328.   int overlaps_vertically();
  1329.   int overlaps_horizontally();
  1330.   void ascii_print(ascii_output_file *);
  1331.   void asciify(macro *);
  1332.   int character_type();
  1333.   int same(node *);
  1334.   const char *type();
  1335. };
  1336.  
  1337. glyph_node *glyph_node::free_list = 0;
  1338.  
  1339. class ligature_node : public glyph_node {
  1340.   node *n1;
  1341.   node *n2;
  1342. #ifdef STORE_WIDTH
  1343.   ligature_node(charinfo *, tfont *, hunits, node *gn1, node *gn2, node *x = 0);
  1344. #endif
  1345. public:
  1346.   void *operator new(size_t);
  1347.   void operator delete(void *);
  1348.   ligature_node(charinfo *, tfont *, node *gn1, node *gn2, node *x = 0);
  1349.   ~ligature_node();
  1350.   node *copy();
  1351.   node *add_self(node *, hyphen_list **);
  1352.   hyphen_list *get_hyphen_list(hyphen_list *ss = 0);
  1353.   void ascii_print(ascii_output_file *);
  1354.   void asciify(macro *);
  1355.   int same(node *);
  1356.   const char *type();
  1357. };
  1358.  
  1359. class kern_pair_node : public node {
  1360.   hunits amount;
  1361.   node *n1;
  1362.   node *n2;
  1363. public:
  1364.   kern_pair_node(hunits n, node *first, node *second, node *x = 0);
  1365.   ~kern_pair_node();
  1366.   node *copy();
  1367.   node *merge_glyph_node(glyph_node *);
  1368.   node *add_self(node *, hyphen_list **);
  1369.   hyphen_list *get_hyphen_list(hyphen_list *ss = 0);
  1370.   node *add_discretionary_hyphen();
  1371.   hunits width();
  1372.   node *last_char_node();
  1373.   hunits italic_correction();
  1374.   hunits subscript_correction();
  1375.   void tprint(troff_output_file *);
  1376.   hyphenation_type get_hyphenation_type();
  1377.   int ends_sentence();
  1378.   void ascii_print(ascii_output_file *);
  1379.   void asciify(macro *);
  1380.   int same(node *);
  1381.   const char *type();
  1382. };
  1383.  
  1384. class dbreak_node : public node {
  1385.   node *none;
  1386.   node *pre;
  1387.   node *post;
  1388. public:
  1389.   dbreak_node(node *n, node *p, node *x = 0);
  1390.   ~dbreak_node();
  1391.   node *copy();
  1392.   node *merge_glyph_node(glyph_node *);
  1393.   node *add_discretionary_hyphen();
  1394.   hunits width();
  1395.   node *last_char_node();
  1396.   hunits italic_correction();
  1397.   hunits subscript_correction();
  1398.   void tprint(troff_output_file *);
  1399.   breakpoint *get_breakpoints(hunits width, int ns, breakpoint *rest = 0,
  1400.                   int is_inner = 0);
  1401.   int nbreaks();
  1402.   int ends_sentence();
  1403.   void split(int, node **, node **);
  1404.   hyphenation_type get_hyphenation_type();
  1405.   void ascii_print(ascii_output_file *);
  1406.   void asciify(macro *);
  1407.   int same(node *);
  1408.   const char *type();
  1409. };
  1410.  
  1411. void *glyph_node::operator new(size_t n)
  1412. {
  1413.   assert(n == sizeof(glyph_node));
  1414.   if (!free_list) {
  1415.     const int BLOCK = 1024;
  1416.     free_list = (glyph_node *)new char[sizeof(glyph_node)*BLOCK];
  1417.     for (int i = 0; i < BLOCK - 1; i++)
  1418.       free_list[i].next = free_list + i + 1;
  1419.     free_list[BLOCK-1].next = 0;
  1420.   }
  1421.   glyph_node *p = free_list;
  1422.   free_list = (glyph_node *)(free_list->next);
  1423.   p->next = 0;
  1424.   return p;
  1425. }
  1426.  
  1427. void *ligature_node::operator new(size_t n)
  1428. {
  1429.   return new char[n];
  1430. }
  1431.  
  1432. void glyph_node::operator delete(void *p)
  1433. {
  1434.   if (p) {
  1435.     ((glyph_node *)p)->next = free_list;
  1436.     free_list = (glyph_node *)p;
  1437.   }
  1438. }
  1439.  
  1440. void ligature_node::operator delete(void *p)
  1441. {
  1442.   delete p;
  1443. }
  1444.  
  1445. glyph_node::glyph_node(charinfo *c, tfont *t, node *x)
  1446.      : ci(c), tf(t), node(x)
  1447. {
  1448. #ifdef STORE_WIDTH
  1449.   wid = tf->get_width(ci);
  1450. #endif
  1451. }
  1452.  
  1453. #ifdef STORE_WIDTH
  1454. glyph_node::glyph_node(charinfo *c, tfont *t, hunits w, node *x)
  1455.      : ci(c), tf(t), wid(w), node(x)
  1456. {
  1457. }
  1458. #endif
  1459.  
  1460. node *glyph_node::copy()
  1461. {
  1462. #ifdef STORE_WIDTH
  1463.   return new glyph_node(ci, tf, wid);
  1464. #else
  1465.   return new glyph_node(ci, tf);
  1466. #endif
  1467. }
  1468.  
  1469. node *glyph_node::merge_self(node *nd)
  1470. {
  1471.   return nd->merge_glyph_node(this);
  1472. }
  1473.  
  1474. int glyph_node::character_type()
  1475. {
  1476.   return tf->get_character_type(ci);
  1477. }
  1478.  
  1479. node *glyph_node::add_self(node *n, hyphen_list **p)
  1480. {
  1481.   assert(ci->get_hyphenation_code() == (*p)->hyphenation_code);
  1482.   next = 0;
  1483.   node *nn;
  1484.   if (n == 0 || (nn = n->merge_glyph_node(this)) == 0) {
  1485.     next = n;
  1486.     nn = this;
  1487.   }
  1488.   if ((*p)->hyphen)
  1489.     nn = nn->add_discretionary_hyphen();
  1490.   hyphen_list *pp = *p;
  1491.   *p = (*p)->next;
  1492.   delete pp;
  1493.   return nn;
  1494. }
  1495.  
  1496. int glyph_node::overlaps_horizontally()
  1497. {
  1498.   return ci->overlaps_horizontally();
  1499. }
  1500.  
  1501. int glyph_node::overlaps_vertically()
  1502. {
  1503.   return ci->overlaps_vertically();
  1504. }
  1505.  
  1506. units glyph_node::size()
  1507. {
  1508.   return tf->get_size().to_units();
  1509. }
  1510.  
  1511. hyphen_list *glyph_node::get_hyphen_list(hyphen_list *tail)
  1512. {
  1513.   return new hyphen_list(ci->get_hyphenation_code(), tail);
  1514. }
  1515.  
  1516.  
  1517. tfont *node::get_tfont()
  1518. {
  1519.   return 0;
  1520. }
  1521.  
  1522. tfont *glyph_node::get_tfont()
  1523. {
  1524.   return tf;
  1525. }
  1526.  
  1527. node *node::merge_glyph_node(glyph_node * /*gn*/)
  1528. {
  1529.   return 0;
  1530. }
  1531.  
  1532. node *glyph_node::merge_glyph_node(glyph_node *gn)
  1533. {
  1534.   if (tf == gn->tf) {
  1535.     charinfo *lig;
  1536.     if ((lig = tf->get_lig(ci, gn->ci)) != 0) {
  1537.       node *next1 = next;
  1538.       next = 0;
  1539.       return new ligature_node(lig, tf, this, gn, next1);
  1540.     }
  1541.     hunits kern;
  1542.     if (tf->get_kern(ci, gn->ci, &kern)) {
  1543.       node *next1 = next;
  1544.       next = 0;
  1545.       return new kern_pair_node(kern, this, gn, next1);
  1546.     }
  1547.   }
  1548.   return 0;
  1549. }
  1550.  
  1551. #ifdef STORE_WIDTH
  1552. inline
  1553. #endif
  1554. hunits glyph_node::width()
  1555. {
  1556. #ifdef STORE_WIDTH
  1557.   return wid;
  1558. #else
  1559.   return tf->get_width(ci);
  1560. #endif
  1561. }
  1562.  
  1563. node *glyph_node::last_char_node()
  1564. {
  1565.   return this;
  1566. }
  1567.  
  1568. void glyph_node::vertical_extent(vunits *min, vunits *max)
  1569. {
  1570.   *min = -tf->get_char_height(ci);
  1571.   *max = tf->get_char_depth(ci);
  1572. }
  1573.  
  1574. hunits glyph_node::skew()
  1575. {
  1576.   return tf->get_char_skew(ci);
  1577. }
  1578.  
  1579. hunits glyph_node::subscript_correction()
  1580. {
  1581.   return tf->get_subscript_correction(ci);
  1582. }
  1583.  
  1584. hunits glyph_node::italic_correction()
  1585. {
  1586.   return tf->get_italic_correction(ci);
  1587. }
  1588.  
  1589. hunits glyph_node::left_italic_correction()
  1590. {
  1591.   return tf->get_left_italic_correction(ci);
  1592. }
  1593.  
  1594. hyphenation_type glyph_node::get_hyphenation_type()
  1595. {
  1596.   return HYPHEN_MIDDLE;
  1597. }
  1598.  
  1599. int glyph_node::ends_sentence()
  1600. {
  1601.   if (ci->ends_sentence())
  1602.     return 1;
  1603.   else if (ci->transparent())
  1604.     return 2;
  1605.   else
  1606.     return 0;
  1607. }
  1608.  
  1609. void glyph_node::ascii_print(ascii_output_file *ascii)
  1610. {
  1611.   unsigned char c = ci->get_ascii_code();
  1612.   if (c != 0)
  1613.     ascii->outc(c);
  1614.   else
  1615.     ascii->outs(ci->nm.contents());
  1616. }
  1617.  
  1618. ligature_node::ligature_node(charinfo *c, tfont *t, 
  1619.                  node *gn1, node *gn2, node *x)
  1620.      : glyph_node(c, t, x), n1(gn1), n2(gn2)
  1621. {
  1622. }
  1623.  
  1624. #ifdef STORE_WIDTH
  1625. ligature_node::ligature_node(charinfo *c, tfont *t, hunits w,
  1626.                    node *gn1, node *gn2, node *x)
  1627.      : glyph_node(c, t, w, x), n1(gn1), n2(gn2)
  1628. {
  1629. }
  1630. #endif
  1631.  
  1632. ligature_node::~ligature_node()
  1633. {
  1634.   delete n1;
  1635.   delete n2;
  1636. }
  1637.  
  1638. node *ligature_node::copy()
  1639. {
  1640. #ifdef STORE_WIDTH
  1641.   return new ligature_node(ci, tf, wid, n1->copy(), n2->copy());
  1642. #else
  1643.   return new ligature_node(ci, tf, n1->copy(), n2->copy());
  1644. #endif
  1645. }
  1646.  
  1647. void ligature_node::ascii_print(ascii_output_file *ascii)
  1648. {
  1649.   n1->ascii_print(ascii);
  1650.   n2->ascii_print(ascii);
  1651. }
  1652.  
  1653. hyphen_list *ligature_node::get_hyphen_list(hyphen_list *tail)
  1654. {
  1655.   return n1->get_hyphen_list(n2->get_hyphen_list(tail));
  1656. }
  1657.  
  1658. node *ligature_node::add_self(node *n, hyphen_list **p)
  1659. {
  1660.   n = n1->add_self(n, p);
  1661.   n = n2->add_self(n, p);
  1662.   n1 = n2 = 0;
  1663.   delete this;
  1664.   return n;
  1665. }
  1666.  
  1667. kern_pair_node::kern_pair_node(hunits n, node *first, node *second, node *x)
  1668.      : node(x), n1(first), n2(second), amount(n)
  1669. {
  1670. }
  1671.  
  1672. dbreak_node::dbreak_node(node *n, node *p, node *x) 
  1673.      : node(x), none(n), pre(p), post(0)
  1674. {
  1675. }
  1676.  
  1677. node *dbreak_node::merge_glyph_node(glyph_node *gn)
  1678. {
  1679.   glyph_node *gn2 = (glyph_node *)gn->copy();
  1680.   node *new_none = none ? none->merge_glyph_node(gn) : 0;
  1681.   node *new_post = post ? post->merge_glyph_node(gn2) : 0;
  1682.   if (new_none == 0 && new_post == 0) {
  1683.     delete gn2;
  1684.     return 0;
  1685.   }
  1686.   if (new_none != 0)
  1687.     none = new_none;
  1688.   else {
  1689.     gn->next = none;
  1690.     none = gn;
  1691.   }
  1692.   if (new_post != 0)
  1693.     post = new_post;
  1694.   else {
  1695.     gn2->next = post;
  1696.     post = gn2;
  1697.   }
  1698.   return this;
  1699. }
  1700.  
  1701. node *kern_pair_node::merge_glyph_node(glyph_node *gn)
  1702. {
  1703.   node *nd = n2->merge_glyph_node(gn);
  1704.   if (nd == 0)
  1705.     return 0;
  1706.   n2 = nd;
  1707.   nd = n2->merge_self(n1);
  1708.   if (nd) {
  1709.     nd->next = next;
  1710.     n1 = 0;
  1711.     n2 = 0;
  1712.     delete this;
  1713.     return nd;
  1714.   }
  1715.   return this;
  1716. }
  1717.  
  1718.  
  1719. hunits kern_pair_node::italic_correction()
  1720. {
  1721.   return n2->italic_correction();
  1722. }
  1723.  
  1724. hunits kern_pair_node::subscript_correction()
  1725. {
  1726.   return n2->subscript_correction();
  1727. }
  1728.  
  1729. node *kern_pair_node::add_discretionary_hyphen()
  1730. {
  1731.   tfont *tf = n2->get_tfont();
  1732.   if (tf) {
  1733.     if (tf->contains(soft_hyphen_char)) {
  1734.       node *next1 = next;
  1735.       next = 0;
  1736.       node *n = copy();
  1737.       glyph_node *gn = new glyph_node(soft_hyphen_char, tf);
  1738.       node *nn = n->merge_glyph_node(gn);
  1739.       if (nn == 0) {
  1740.     gn->next = n;
  1741.     nn = gn;
  1742.       }
  1743.       return new dbreak_node(this, nn, next1);
  1744.     }
  1745.   }
  1746.   return this;
  1747. }
  1748.  
  1749.  
  1750. kern_pair_node::~kern_pair_node()
  1751. {
  1752.   if (n1 != 0)
  1753.     delete n1;
  1754.   if (n2 != 0)
  1755.     delete n2;
  1756. }
  1757.  
  1758. dbreak_node::~dbreak_node()
  1759. {
  1760.   delete_node_list(pre);
  1761.   delete_node_list(post);
  1762.   delete_node_list(none);
  1763. }
  1764.  
  1765. node *kern_pair_node::copy()
  1766. {
  1767.   return new kern_pair_node(amount, n1->copy(), n2->copy());
  1768. }
  1769.  
  1770. node *copy_node_list(node *n)
  1771. {
  1772.   node *p = 0;
  1773.   while (n != 0) {
  1774.     node *nn = n->copy();
  1775.     nn->next = p;
  1776.     p = nn;
  1777.     n = n->next;
  1778.   }
  1779.   while (p != 0) {
  1780.     node *pp = p->next;
  1781.     p->next = n;
  1782.     n = p;
  1783.     p = pp;
  1784.   }
  1785.   return n;
  1786. }
  1787.  
  1788. void delete_node_list(node *n)
  1789. {
  1790.   while (n != 0) {
  1791.     node *tem = n;
  1792.     n = n->next;
  1793.     delete tem;
  1794.   }
  1795. }
  1796.  
  1797. node *dbreak_node::copy()
  1798. {
  1799.   dbreak_node *p = new dbreak_node(copy_node_list(none), copy_node_list(pre));
  1800.   p->post = copy_node_list(post);
  1801.   return p;
  1802. }
  1803.  
  1804. hyphen_list *node::get_hyphen_list(hyphen_list *tail)
  1805. {
  1806.   return tail;
  1807. }
  1808.  
  1809.  
  1810. hyphen_list *kern_pair_node::get_hyphen_list(hyphen_list *tail)
  1811. {
  1812.   return n1->get_hyphen_list(n2->get_hyphen_list(tail));
  1813. }
  1814.  
  1815. class hyphen_inhibitor_node : public node {
  1816. public:
  1817.   hyphen_inhibitor_node(node *nd = 0);
  1818.   node *copy();
  1819.   int same(node *);
  1820.   const char *type();
  1821.   hyphenation_type get_hyphenation_type();
  1822. };
  1823.  
  1824. hyphen_inhibitor_node::hyphen_inhibitor_node(node *nd) : node(nd)
  1825. {
  1826. }
  1827.  
  1828. node *hyphen_inhibitor_node::copy()
  1829. {
  1830.   return new hyphen_inhibitor_node;
  1831. }
  1832.  
  1833. int hyphen_inhibitor_node::same(node *)
  1834. {
  1835.   return 1;
  1836. }
  1837.  
  1838. const char *hyphen_inhibitor_node::type()
  1839. {
  1840.   return "hyphen_inhibitor_node";
  1841. }
  1842.  
  1843. hyphenation_type hyphen_inhibitor_node::get_hyphenation_type()
  1844. {
  1845.   return HYPHEN_INHIBIT;
  1846. }
  1847.  
  1848. /* add_discretionary_hyphen methods */
  1849.  
  1850. node *dbreak_node::add_discretionary_hyphen()
  1851. {
  1852.   if (post)
  1853.     post = post->add_discretionary_hyphen();
  1854.   if (none)
  1855.     none = none->add_discretionary_hyphen();
  1856.   return this;
  1857. }
  1858.  
  1859.  
  1860. node *node::add_discretionary_hyphen()
  1861. {
  1862.   tfont *tf = get_tfont();
  1863.   if (!tf)
  1864.     return new hyphen_inhibitor_node(this);
  1865.   if (tf->contains(soft_hyphen_char)) {
  1866.     node *next1 = next;
  1867.     next = 0;
  1868.     node *n = copy();
  1869.     glyph_node *gn = new glyph_node(soft_hyphen_char, tf);
  1870.     node *n1 = n->merge_glyph_node(gn);
  1871.     if (n1 == 0) {
  1872.       gn->next = n;
  1873.       n1 = gn;
  1874.     }
  1875.     return new dbreak_node(this, n1, next1);
  1876.   }
  1877.   return this;
  1878. }
  1879.  
  1880.  
  1881. node *node::merge_self(node *)
  1882. {
  1883.   return 0;
  1884. }
  1885.  
  1886. node *node::add_self(node *n, hyphen_list ** /*p*/)
  1887. {
  1888.   next = n;
  1889.   return this;
  1890. }
  1891.  
  1892. node *kern_pair_node::add_self(node *n, hyphen_list **p)
  1893. {
  1894.   n = n1->add_self(n, p);
  1895.   n = n2->add_self(n, p);
  1896.   n1 = n2 = 0;
  1897.   delete this;
  1898.   return n;
  1899. }
  1900.  
  1901.  
  1902. hunits node::width()
  1903. {
  1904.   return H0;
  1905. }
  1906.  
  1907. node *node::last_char_node()
  1908. {
  1909.   return 0;
  1910. }
  1911.  
  1912. hunits hmotion_node::width()
  1913. {
  1914.   return n;
  1915. }
  1916.  
  1917. units node::size()
  1918. {
  1919.   return points_to_units(10);
  1920. }
  1921.  
  1922. hunits kern_pair_node::width()
  1923. {
  1924.   return n1->width() + n2->width() + amount;
  1925. }
  1926.  
  1927. node *kern_pair_node::last_char_node()
  1928. {
  1929.   node *nd = n2->last_char_node();
  1930.   if (nd)
  1931.     return nd;
  1932.   return n1->last_char_node();
  1933. }
  1934.  
  1935. hunits dbreak_node::width()
  1936. {
  1937.   hunits x = H0;
  1938.   for (node *n = none; n != 0; n = n->next)
  1939.     x += n->width();
  1940.   return x;
  1941. }
  1942.  
  1943. node *dbreak_node::last_char_node()
  1944. {
  1945.   for (node *n = none; n; n = n->next) {
  1946.     node *last = n->last_char_node();
  1947.     if (last)
  1948.       return last;
  1949.   }
  1950.   return 0;
  1951. }
  1952.  
  1953. hunits dbreak_node::italic_correction()
  1954. {
  1955.   return none ? none->italic_correction() : H0;
  1956. }
  1957.  
  1958. hunits dbreak_node::subscript_correction()
  1959. {
  1960.   return none ? none->subscript_correction() : H0;
  1961. }
  1962.  
  1963. class italic_corrected_node : public node {
  1964.   node *n;
  1965.   hunits x;
  1966. public:
  1967.   italic_corrected_node(node *, hunits, node * = 0);
  1968.   ~italic_corrected_node();
  1969.   node *copy();
  1970.   void ascii_print(ascii_output_file *);
  1971.   void asciify(macro *m);
  1972.   hunits width();
  1973.   node *last_char_node();
  1974.   void vertical_extent(vunits *, vunits *);
  1975.   int ends_sentence();
  1976.   int overlaps_horizontally();
  1977.   int overlaps_vertically();
  1978.   int same(node *);
  1979.   hyphenation_type get_hyphenation_type();
  1980.   tfont *get_tfont();
  1981.   hyphen_list *get_hyphen_list(hyphen_list *ss = 0);
  1982.   int character_type();
  1983.   void tprint(troff_output_file *);
  1984.   hunits subscript_correction();
  1985.   hunits skew();
  1986.   node *add_self(node *, hyphen_list **);
  1987.   const char *type();
  1988. };
  1989.  
  1990. node *node::add_italic_correction(hunits *width)
  1991. {
  1992.   hunits ic = italic_correction();
  1993.   if (ic.is_zero())
  1994.     return this;
  1995.   else {
  1996.     node *next1 = next;
  1997.     next = 0;
  1998.     *width += ic;
  1999.     return new italic_corrected_node(this, ic, next1);
  2000.   }
  2001. }
  2002.  
  2003. italic_corrected_node::italic_corrected_node(node *nn, hunits xx, node *p)
  2004. : n(nn), x(xx), node(p)
  2005. {
  2006.   assert(n != 0);
  2007. }
  2008.  
  2009. italic_corrected_node::~italic_corrected_node()
  2010. {
  2011.   delete n;
  2012. }
  2013.  
  2014. node *italic_corrected_node::copy()
  2015. {
  2016.   return new italic_corrected_node(n->copy(), x);
  2017. }
  2018.  
  2019. hunits italic_corrected_node::width()
  2020. {
  2021.   return n->width() + x;
  2022. }
  2023.  
  2024. void italic_corrected_node::vertical_extent(vunits *min, vunits *max)
  2025. {
  2026.   n->vertical_extent(min, max);
  2027. }
  2028.  
  2029. void italic_corrected_node::tprint(troff_output_file *out)
  2030. {
  2031.   n->tprint(out);
  2032.   out->right(x);
  2033. }
  2034.  
  2035. hunits italic_corrected_node::skew()
  2036. {
  2037.   return n->skew() - x/2;
  2038. }
  2039.  
  2040. hunits italic_corrected_node::subscript_correction()
  2041. {
  2042.   return n->subscript_correction() - x;
  2043. }
  2044.  
  2045. void italic_corrected_node::ascii_print(ascii_output_file *out)
  2046. {
  2047.   n->ascii_print(out);
  2048. }
  2049.  
  2050. int italic_corrected_node::ends_sentence()
  2051. {
  2052.   return n->ends_sentence();
  2053. }
  2054.  
  2055. int italic_corrected_node::overlaps_horizontally()
  2056. {
  2057.   return n->overlaps_horizontally();
  2058. }
  2059.  
  2060. int italic_corrected_node::overlaps_vertically()
  2061. {
  2062.   return n->overlaps_vertically();
  2063. }
  2064.  
  2065. node *italic_corrected_node::last_char_node()
  2066. {
  2067.   return n->last_char_node();
  2068. }
  2069.  
  2070. tfont *italic_corrected_node::get_tfont()
  2071. {
  2072.   return n->get_tfont();
  2073. }
  2074.  
  2075. hyphenation_type italic_corrected_node::get_hyphenation_type()
  2076. {
  2077.   return n->get_hyphenation_type();
  2078. }
  2079.  
  2080. node *italic_corrected_node::add_self(node *nd, hyphen_list **p)
  2081. {
  2082.   nd = n->add_self(nd, p);
  2083.   hunits not_interested;
  2084.   nd = nd->add_italic_correction(¬_interested);
  2085.   n = 0;
  2086.   delete this;
  2087.   return nd;
  2088. }
  2089.  
  2090. hyphen_list *italic_corrected_node::get_hyphen_list(hyphen_list *tail)
  2091. {
  2092.   return n->get_hyphen_list(tail);
  2093. }
  2094.  
  2095. int italic_corrected_node::character_type()
  2096. {
  2097.   return n->character_type();
  2098. }
  2099.  
  2100. class break_char_node : public node {
  2101.   node *ch;
  2102.   char break_code;
  2103. public:
  2104.   break_char_node(node *, int, node * = 0);
  2105.   ~break_char_node();
  2106.   node *copy();
  2107.   hunits width();
  2108.   vunits vertical_width();
  2109.   node *last_char_node();
  2110.   int character_type();
  2111.   int ends_sentence();
  2112.   node *add_self(node *, hyphen_list **);
  2113.   hyphen_list *get_hyphen_list(hyphen_list *s = 0);
  2114.   void tprint(troff_output_file *);
  2115.   void zero_width_tprint(troff_output_file *);
  2116.   void ascii_print(ascii_output_file *);
  2117.   void asciify(macro *m);
  2118.   hyphenation_type get_hyphenation_type();
  2119.   int overlaps_vertically();
  2120.   int overlaps_horizontally();
  2121.   units size();
  2122.   tfont *get_tfont();
  2123.   int same(node *);
  2124.   const char *type();
  2125. };
  2126.  
  2127. break_char_node::break_char_node(node *n, int c, node *x)
  2128. : node(x), ch(n), break_code(c)
  2129. {
  2130. }
  2131.  
  2132. break_char_node::~break_char_node()
  2133. {
  2134.   delete ch;
  2135. }
  2136.  
  2137. node *break_char_node::copy()
  2138. {
  2139.   return new break_char_node(ch->copy(), break_code);
  2140. }
  2141.  
  2142. hunits break_char_node::width()
  2143. {
  2144.   return ch->width();
  2145. }
  2146.  
  2147. vunits break_char_node::vertical_width()
  2148. {
  2149.   return ch->vertical_width();
  2150. }
  2151.  
  2152. node *break_char_node::last_char_node()
  2153. {
  2154.   return ch->last_char_node();
  2155. }
  2156.  
  2157. int break_char_node::character_type()
  2158. {
  2159.   return ch->character_type();
  2160. }
  2161.  
  2162. int break_char_node::ends_sentence()
  2163. {
  2164.   return ch->ends_sentence();
  2165. }
  2166.  
  2167. node *break_char_node::add_self(node *n, hyphen_list **p)
  2168. {
  2169.   assert((*p)->hyphenation_code == 0);
  2170.   if ((*p)->breakable && (break_code & 1)) {
  2171.     n = new space_node(H0, n);
  2172.     n->freeze_space();
  2173.   }
  2174.   next = n;
  2175.   n = this;
  2176.   if ((*p)->breakable && (break_code & 2)) {
  2177.     n = new space_node(H0, n);
  2178.     n->freeze_space();
  2179.   }
  2180.   hyphen_list *pp = *p;
  2181.   *p = (*p)->next;
  2182.   delete pp;
  2183.   return n;
  2184. }
  2185.  
  2186. hyphen_list *break_char_node::get_hyphen_list(hyphen_list *tail)
  2187. {
  2188.   return new hyphen_list(0, tail);
  2189. }
  2190.  
  2191. hyphenation_type break_char_node::get_hyphenation_type()
  2192. {
  2193.   return HYPHEN_MIDDLE;
  2194. }
  2195.  
  2196. void break_char_node::ascii_print(ascii_output_file *ascii)
  2197. {
  2198.   ch->ascii_print(ascii);
  2199. }
  2200.  
  2201. int break_char_node::overlaps_vertically()
  2202. {
  2203.   return ch->overlaps_vertically();
  2204. }
  2205.  
  2206. int break_char_node::overlaps_horizontally()
  2207. {
  2208.   return ch->overlaps_horizontally();
  2209. }
  2210.  
  2211. units break_char_node::size()
  2212. {
  2213.   return ch->size();
  2214. }
  2215.  
  2216. tfont *break_char_node::get_tfont()
  2217. {
  2218.   return ch->get_tfont();
  2219. }
  2220.  
  2221. node *extra_size_node::copy()
  2222.   return new extra_size_node(n); 
  2223. }
  2224.  
  2225. node *vertical_size_node::copy()
  2226.   return new vertical_size_node(n); 
  2227. }
  2228.  
  2229. node *hmotion_node::copy()
  2230.   return new hmotion_node(n);
  2231. }
  2232.  
  2233. node *space_char_hmotion_node::copy()
  2234. {
  2235.   return new space_char_hmotion_node(n);
  2236. }
  2237.  
  2238. node *vmotion_node::copy()
  2239.   return new vmotion_node(n);
  2240. }
  2241.   
  2242. node *dummy_node::copy()
  2243.   return new dummy_node;
  2244. }
  2245.  
  2246. node *transparent_dummy_node::copy()
  2247.   return new transparent_dummy_node;
  2248. }
  2249.  
  2250. hline_node::~hline_node()
  2251. {
  2252.   if (n)
  2253.     delete n;
  2254. }
  2255.  
  2256. node *hline_node::copy()
  2257. {
  2258.   return new hline_node(x, n ? n->copy() : 0);
  2259. }
  2260.  
  2261. hunits hline_node::width()
  2262. {
  2263.   return x < H0 ? H0 : x;
  2264. }
  2265.  
  2266.  
  2267. vline_node::~vline_node()
  2268. {
  2269.   if (n)
  2270.     delete n;
  2271. }
  2272.  
  2273. node *vline_node::copy()
  2274. {
  2275.   return new vline_node(x, n ? n->copy() : 0);
  2276. }
  2277.  
  2278. hunits vline_node::width()
  2279. {
  2280.   return n == 0 ? H0 : n->width();
  2281. }
  2282.  
  2283.  
  2284. zero_width_node::zero_width_node(node *nd) : n(nd)
  2285. {
  2286. }
  2287.  
  2288. zero_width_node::~zero_width_node()
  2289. {
  2290.   delete_node_list(n);
  2291. }
  2292.  
  2293. node *zero_width_node::copy()
  2294. {
  2295.   return new zero_width_node(copy_node_list(n));
  2296. }
  2297.  
  2298. int node_list_character_type(node *p)
  2299. {
  2300.   int t = 0;
  2301.   for (; p; p = p->next)
  2302.     t |= p->character_type();
  2303.   return t;
  2304. }
  2305.  
  2306. int zero_width_node::character_type()
  2307. {
  2308.   return node_list_character_type(n);
  2309. }
  2310.  
  2311. void node_list_vertical_extent(node *p, vunits *min, vunits *max)
  2312. {
  2313.   *min = V0;
  2314.   *max = V0;
  2315.   vunits cur_vpos = V0;
  2316.   vunits v1, v2;
  2317.   for (; p; p = p->next) {
  2318.     p->vertical_extent(&v1, &v2);
  2319.     v1 += cur_vpos;
  2320.     if (v1 < *min)
  2321.       *min = v1;
  2322.     v2 += cur_vpos;
  2323.     if (v2 > *max)
  2324.       *max = v2;
  2325.     cur_vpos += p->vertical_width();
  2326.   }
  2327. }
  2328.  
  2329. void zero_width_node::vertical_extent(vunits *min, vunits *max)
  2330. {
  2331.   node_list_vertical_extent(n, min, max);
  2332. }
  2333.  
  2334. overstrike_node::overstrike_node() : max_width(H0), list(0)
  2335. {
  2336. }
  2337.  
  2338. overstrike_node::~overstrike_node()
  2339. {
  2340.   delete_node_list(list);
  2341. }
  2342.  
  2343. node *overstrike_node::copy()
  2344. {
  2345.   overstrike_node *on = new overstrike_node;
  2346.   for (node *tem = list; tem; tem = tem->next)
  2347.     on->overstrike(tem->copy());
  2348.   return on;
  2349. }
  2350.  
  2351. void overstrike_node::overstrike(node *n)
  2352. {
  2353.   if (n == 0)
  2354.     return;
  2355.   hunits w = n->width();
  2356.   if (w > max_width)
  2357.     max_width = w;
  2358.   for (node **p = &list; *p; p = &(*p)->next)
  2359.     ;
  2360.   n->next = 0;
  2361.   *p = n;
  2362. }
  2363.  
  2364. hunits overstrike_node::width()
  2365. {
  2366.   return max_width;
  2367. }
  2368.  
  2369. bracket_node::bracket_node() : max_width(H0), list(0)
  2370. {
  2371. }
  2372.  
  2373. bracket_node::~bracket_node()
  2374. {
  2375.   delete_node_list(list);
  2376. }
  2377.  
  2378. node *bracket_node::copy()
  2379. {
  2380.   bracket_node *on = new bracket_node;
  2381.   for (node *tem = list; tem; tem = tem->next)
  2382.     on->bracket(tem->copy());
  2383.   return on;
  2384. }
  2385.  
  2386.  
  2387. void bracket_node::bracket(node *n)
  2388. {
  2389.   if (n == 0)
  2390.     return;
  2391.   hunits w = n->width();
  2392.   if (w > max_width)
  2393.     max_width = w;
  2394.   n->next = list;
  2395.   list = n;
  2396. }
  2397.  
  2398. hunits bracket_node::width()
  2399. {
  2400.   return max_width;
  2401. }
  2402.  
  2403. int node::nspaces()
  2404. {
  2405.   return 0;
  2406. }
  2407.  
  2408. int node::merge_space(hunits)
  2409. {
  2410.   return 0;
  2411. }
  2412.  
  2413. #if 0
  2414. space_node *space_node::free_list = 0;
  2415.  
  2416. void *space_node::operator new(size_t n)
  2417. {
  2418.   assert(n == sizeof(space_node));
  2419.   if (!free_list) {
  2420.     free_list = (space_node *)new char[sizeof(space_node)*BLOCK];
  2421.     for (int i = 0; i < BLOCK - 1; i++)
  2422.       free_list[i].next = free_list + i + 1;
  2423.     free_list[BLOCK-1].next = 0;
  2424.   }
  2425.   space_node *p = free_list;
  2426.   free_list = (space_node *)(free_list->next);
  2427.   p->next = 0;
  2428.   return p;
  2429. }
  2430.  
  2431. inline void space_node::operator delete(void *p)
  2432. {
  2433.   if (p) {
  2434.     ((space_node *)p)->next = free_list;
  2435.     free_list = (space_node *)p;
  2436.   }
  2437. }
  2438. #endif
  2439.  
  2440. space_node::space_node(hunits nn, node *p) : node(p), n(nn), set(0)
  2441. {
  2442. }
  2443.  
  2444. space_node::space_node(hunits nn, int s, node *p) : node(p), n(nn), set(s)
  2445. {
  2446. }
  2447.  
  2448. #if 0
  2449. space_node::~space_node()
  2450. {
  2451. }
  2452. #endif
  2453.  
  2454. node *space_node::copy()
  2455. {
  2456.   return new space_node(n, set);
  2457. }
  2458.  
  2459. int space_node::nspaces()
  2460. {
  2461.   return set ? 0 : 1;
  2462. }
  2463.  
  2464. int space_node::merge_space(hunits h)
  2465. {
  2466.   n += h;
  2467.   return 1;
  2468. }
  2469.  
  2470. hunits space_node::width()
  2471. {
  2472.   return n;
  2473. }
  2474.  
  2475. void node::spread_space(int*, hunits*)
  2476. {
  2477. }
  2478.  
  2479. void space_node::spread_space(int *nspaces, hunits *desired_space)
  2480. {
  2481.   if (!set) {
  2482.     assert(*nspaces > 0);
  2483.     if (*nspaces == 1) {
  2484.       n += *desired_space;
  2485.       *desired_space = H0;
  2486.     }
  2487.     else {
  2488.       hunits extra = *desired_space / *nspaces;
  2489.       *desired_space -= extra;
  2490.       n += extra;
  2491.     }
  2492.     *nspaces -= 1;
  2493.     set = 1;
  2494.   }
  2495. }
  2496.  
  2497. void node::freeze_space()
  2498. {
  2499. }
  2500.  
  2501. void space_node::freeze_space()
  2502. {
  2503.   set = 1;
  2504. }
  2505.  
  2506. diverted_space_node::diverted_space_node(vunits d, node *p)
  2507. : node(p), n(d)
  2508. {
  2509. }
  2510.  
  2511. node *diverted_space_node::copy()
  2512. {
  2513.   return new diverted_space_node(n);
  2514. }
  2515.  
  2516. diverted_copy_file_node::diverted_copy_file_node(symbol s, node *p)
  2517. : node(p), filename(s)
  2518. {
  2519. }
  2520.  
  2521. node *diverted_copy_file_node::copy()
  2522. {
  2523.   return new diverted_copy_file_node(filename);
  2524. }
  2525.  
  2526. int node::ends_sentence()
  2527. {
  2528.   return 0;
  2529. }
  2530.  
  2531. int kern_pair_node::ends_sentence()
  2532. {
  2533.   switch (n2->ends_sentence()) {
  2534.   case 0:
  2535.     return 0;
  2536.   case 1:
  2537.     return 1;
  2538.   case 2:
  2539.     break;
  2540.   default:
  2541.     assert(0);
  2542.   }
  2543.   return n1->ends_sentence();
  2544. }
  2545.  
  2546. int node_list_ends_sentence(node *n)
  2547. {
  2548.   for (; n != 0; n = n->next)
  2549.     switch (n->ends_sentence()) {
  2550.     case 0:
  2551.       return 0;
  2552.     case 1:
  2553.       return 1;
  2554.     case 2:
  2555.       break;
  2556.     default:
  2557.       assert(0);
  2558.     }
  2559.   return 2;
  2560. }
  2561.  
  2562.     
  2563. int dbreak_node::ends_sentence()
  2564. {
  2565.   return node_list_ends_sentence(none);
  2566. }
  2567.  
  2568.  
  2569. int node::overlaps_horizontally()
  2570. {
  2571.   return 0;
  2572. }
  2573.  
  2574. int node::overlaps_vertically()
  2575. {
  2576.   return 0;
  2577. }
  2578.  
  2579. int node::discardable()
  2580. {
  2581.   return 0;
  2582. }
  2583.  
  2584. int space_node::discardable()
  2585. {
  2586.   return set ? 0 : 1;
  2587. }
  2588.  
  2589.  
  2590. vunits node::vertical_width()
  2591. {
  2592.   return V0;
  2593. }
  2594.  
  2595. vunits vline_node::vertical_width()
  2596. {
  2597.   return x;
  2598. }
  2599.  
  2600. vunits vmotion_node::vertical_width()
  2601. {
  2602.   return n;
  2603. }
  2604.  
  2605. int node::character_type()
  2606. {
  2607.   return 0;
  2608. }
  2609.  
  2610. hunits node::subscript_correction()
  2611. {
  2612.   return H0;
  2613. }
  2614.  
  2615. hunits node::italic_correction()
  2616. {
  2617.   return H0;
  2618. }
  2619.  
  2620. hunits node::left_italic_correction()
  2621. {
  2622.   return H0;
  2623. }
  2624.  
  2625. hunits node::skew()
  2626. {
  2627.   return H0;
  2628. }
  2629.  
  2630.  
  2631. /* vertical_extent methods */
  2632.  
  2633. void node::vertical_extent(vunits *min, vunits *max)
  2634. {
  2635.   vunits v = vertical_width();
  2636.   if (v < V0) {
  2637.     *min = v;
  2638.     *max = V0;
  2639.   }
  2640.   else {
  2641.     *max = v;
  2642.     *min = V0;
  2643.   }
  2644. }
  2645.  
  2646. void vline_node::vertical_extent(vunits *min, vunits *max)
  2647. {
  2648.   if (n == 0)
  2649.     node::vertical_extent(min, max);
  2650.   else {
  2651.     vunits cmin, cmax;
  2652.     n->vertical_extent(&cmin, &cmax);
  2653.     vunits h = n->size();
  2654.     if (x < V0) {
  2655.       if (-x < h) {
  2656.     *min = x;
  2657.     *max = V0;
  2658.       }
  2659.       else {
  2660.     // we print the first character and then move up, so
  2661.     *max = cmax;
  2662.     // we print the last character and then move up h
  2663.     *min = cmin + h;
  2664.     if (*min > V0)
  2665.       *min = V0;
  2666.     *min += x;
  2667.       }
  2668.     }
  2669.     else {
  2670.       if (x < h) {
  2671.     *max = x;
  2672.     *min = V0;
  2673.       }
  2674.       else {
  2675.     // we move down by h and then print the first character, so
  2676.     *min = cmin + h;
  2677.     if (*min > V0)
  2678.       *min = V0;
  2679.     *max = x + cmax;
  2680.       }
  2681.     }
  2682.   }
  2683. }
  2684.  
  2685. /* ascii_print methods */
  2686.  
  2687.  
  2688. static void ascii_print_reverse_node_list(ascii_output_file *ascii, node *n)
  2689. {
  2690.   if (n == 0)
  2691.     return;
  2692.   ascii_print_reverse_node_list(ascii, n->next);
  2693.   n->ascii_print(ascii);
  2694. }
  2695.  
  2696. void dbreak_node::ascii_print(ascii_output_file *ascii)
  2697. {
  2698.   ascii_print_reverse_node_list(ascii, none);
  2699. }
  2700.  
  2701. void kern_pair_node::ascii_print(ascii_output_file *ascii)
  2702. {
  2703.   n1->ascii_print(ascii);
  2704.   n2->ascii_print(ascii);
  2705. }
  2706.  
  2707.  
  2708. void node::ascii_print(ascii_output_file *)
  2709. {
  2710. }
  2711.  
  2712. void space_node::ascii_print(ascii_output_file *ascii)
  2713. {
  2714.   if (!n.is_zero())
  2715.     ascii->outc(' ');
  2716. }
  2717.  
  2718. void hmotion_node::ascii_print(ascii_output_file *ascii)
  2719. {
  2720.   // this is pretty arbitrary
  2721.   if (n >= points_to_units(2))
  2722.     ascii->outc(' ');
  2723. }
  2724.  
  2725. void space_char_hmotion_node::ascii_print(ascii_output_file *ascii)
  2726. {
  2727.   ascii->outc(' ');
  2728. }
  2729.  
  2730. /* asciify methods */
  2731.  
  2732. void node::asciify(macro *m)
  2733. {
  2734.   m->append(this);
  2735. }
  2736.       
  2737. void glyph_node::asciify(macro *m)
  2738. {
  2739.   unsigned char c = ci->get_ascii_code();
  2740.   if (c != 0) {
  2741.     m->append(c);
  2742.     delete this;
  2743.   }
  2744.   else
  2745.     m->append(this);
  2746. }
  2747.  
  2748. void kern_pair_node::asciify(macro *m)
  2749. {
  2750.   n1->asciify(m);
  2751.   n2->asciify(m);
  2752.   n1 = n2 = 0;
  2753.   delete this;
  2754. }
  2755.  
  2756. static void asciify_reverse_node_list(macro *m, node *n)
  2757. {
  2758.   if (n == 0)
  2759.     return;
  2760.   asciify_reverse_node_list(m, n->next);
  2761.   n->asciify(m);
  2762. }
  2763.  
  2764. void dbreak_node::asciify(macro *m)
  2765. {
  2766.   asciify_reverse_node_list(m, none);
  2767.   none = 0;
  2768.   delete this;
  2769. }
  2770.  
  2771. void ligature_node::asciify(macro *m)
  2772. {
  2773.   n1->asciify(m);
  2774.   n2->asciify(m);
  2775.   n1 = n2 = 0;
  2776.   delete this;
  2777. }
  2778.  
  2779. void break_char_node::asciify(macro *m)
  2780. {
  2781.   ch->asciify(m);
  2782.   ch = 0;
  2783.   delete this;
  2784. }
  2785.  
  2786. void italic_corrected_node::asciify(macro *m)
  2787. {
  2788.   n->asciify(m);
  2789.   n = 0;
  2790.   delete this;
  2791. }
  2792.  
  2793. void left_italic_corrected_node::asciify(macro *m)
  2794. {
  2795.   if (n) {
  2796.     n->asciify(m);
  2797.     n = 0;
  2798.   }
  2799.   delete this;
  2800. }
  2801.  
  2802. space_char_hmotion_node::space_char_hmotion_node(hunits i, node *next)
  2803. : hmotion_node(i, next)
  2804. {
  2805. }
  2806.  
  2807. void space_char_hmotion_node::asciify(macro *m)
  2808. {
  2809.   m->append(' ');
  2810.   delete this;
  2811. }
  2812.  
  2813. void line_start_node::asciify(macro *)
  2814. {
  2815.   delete this;
  2816. }
  2817.  
  2818. void vertical_size_node::asciify(macro *)
  2819. {
  2820.   delete this;
  2821. }
  2822.  
  2823. breakpoint *node::get_breakpoints(hunits /*width*/, int /*nspaces*/,
  2824.                   breakpoint *rest, int /*is_inner*/)
  2825. {
  2826.   return rest;
  2827. }
  2828.  
  2829. int node::nbreaks()
  2830. {
  2831.   return 0;
  2832. }
  2833.  
  2834. breakpoint *space_node::get_breakpoints(hunits width, int ns, breakpoint *rest,
  2835.                 int is_inner)
  2836. {
  2837.   if (next->discardable())
  2838.     return rest;
  2839.   breakpoint *bp = new breakpoint;
  2840.   bp->next = rest;
  2841.   bp->width = width;
  2842.   bp->nspaces = ns;
  2843.   bp->hyphenated = 0;
  2844.   if (is_inner) {
  2845.     assert(rest != 0);
  2846.     bp->index = rest->index + 1;
  2847.     bp->nd = rest->nd;
  2848.   }
  2849.   else {
  2850.     bp->nd = this;
  2851.     bp->index = 0;
  2852.   }
  2853.   return bp;
  2854. }
  2855.  
  2856. int space_node::nbreaks()
  2857. {
  2858.   if (next->discardable())
  2859.     return 0;
  2860.   else
  2861.     return 1;
  2862. }
  2863.  
  2864. static breakpoint *node_list_get_breakpoints(node *p, hunits *widthp,
  2865.                          int ns, breakpoint *rest)
  2866. {
  2867.   if (p != 0) {
  2868.     rest = p->get_breakpoints(*widthp, 
  2869.                   ns, 
  2870.                   node_list_get_breakpoints(p->next, widthp, ns,
  2871.                             rest),
  2872.                   1);
  2873.     *widthp += p->width();
  2874.   }
  2875.   return rest;
  2876. }
  2877.  
  2878.  
  2879. breakpoint *dbreak_node::get_breakpoints(hunits width, int ns,
  2880.                      breakpoint *rest, int is_inner)
  2881. {
  2882.   breakpoint *bp = new breakpoint;
  2883.   bp->next = rest;
  2884.   bp->width = width;
  2885.   for (node *tem = pre; tem != 0; tem = tem->next)
  2886.     bp->width += tem->width();
  2887.   bp->nspaces = ns;
  2888.   bp->hyphenated = 1;
  2889.   if (is_inner) {
  2890.     assert(rest != 0);
  2891.     bp->index = rest->index + 1;
  2892.     bp->nd = rest->nd;
  2893.   }
  2894.   else {
  2895.     bp->nd = this;
  2896.     bp->index = 0;
  2897.   }
  2898.   return node_list_get_breakpoints(none, &width, ns, bp);
  2899. }
  2900.  
  2901. int dbreak_node::nbreaks()
  2902. {
  2903.   int i = 1;
  2904.   for (node *tem = none; tem != 0; tem = tem->next)
  2905.     i += tem->nbreaks();
  2906.   return i;
  2907. }
  2908.  
  2909. void node::split(int /*where*/, node ** /*prep*/, node ** /*postp*/)
  2910. {
  2911.   assert(0);
  2912. }
  2913.  
  2914. void space_node::split(int where, node **pre, node **post)
  2915. {
  2916.   assert(where == 0);
  2917.   *pre = next;
  2918.   *post = 0;
  2919.   delete this;
  2920. }
  2921.  
  2922. static void node_list_split(node *p, int *wherep, node **prep, node **postp)
  2923. {
  2924.   if (p == 0)
  2925.     return;
  2926.   int nb = p->nbreaks();
  2927.   node_list_split(p->next, wherep, prep, postp);
  2928.   if (*wherep < 0) {
  2929.     p->next = *postp;
  2930.     *postp = p;
  2931.   }
  2932.   else if (*wherep < nb) {
  2933.     p->next = *prep;
  2934.     p->split(*wherep, prep, postp);
  2935.   }
  2936.   else {
  2937.     p->next = *prep;
  2938.     *prep = p;
  2939.   }
  2940.   *wherep -= nb;
  2941. }
  2942.  
  2943. void dbreak_node::split(int where, node **prep, node **postp)
  2944. {
  2945.   assert(where >= 0);
  2946.   if (where == 0) {
  2947.     *postp = post;
  2948.     post = 0;
  2949.     if (pre == 0)
  2950.       *prep = next;
  2951.     else {
  2952.       for (node *tem = pre; tem->next != 0; tem = tem->next)
  2953.     ;
  2954.       tem->next = next;
  2955.       *prep = pre;
  2956.     }
  2957.     pre = 0;
  2958.     delete this;
  2959.   }
  2960.   else {
  2961.     *prep = next;
  2962.     where -= 1;
  2963.     node_list_split(none, &where, prep, postp);
  2964.     none = 0;
  2965.     delete this;
  2966.   }
  2967. }
  2968.   
  2969.  
  2970. hyphenation_type node::get_hyphenation_type()
  2971. {
  2972.   return HYPHEN_BOUNDARY;
  2973. }
  2974.  
  2975.  
  2976. hyphenation_type dbreak_node::get_hyphenation_type()
  2977. {
  2978.   return HYPHEN_INHIBIT;
  2979. }
  2980.  
  2981. hyphenation_type kern_pair_node::get_hyphenation_type()
  2982. {
  2983.   return HYPHEN_MIDDLE;
  2984. }
  2985.  
  2986. hyphenation_type dummy_node::get_hyphenation_type()
  2987. {
  2988.   return HYPHEN_MIDDLE;
  2989. }
  2990.  
  2991. hyphenation_type transparent_dummy_node::get_hyphenation_type()
  2992. {
  2993.   return HYPHEN_MIDDLE;
  2994. }
  2995.  
  2996. int node::interpret(macro *)
  2997. {
  2998.   return 0;
  2999. }
  3000.  
  3001. special_node::special_node(const macro &m)
  3002. : mac(m)
  3003. {
  3004. }
  3005.  
  3006. int special_node::same(node *n)
  3007. {
  3008.   return mac == ((special_node *)n)->mac;
  3009. }
  3010.  
  3011. const char *special_node::type()
  3012. {
  3013.   return "special_node";
  3014. }
  3015.  
  3016. node *special_node::copy()
  3017. {
  3018.   return new special_node(mac);
  3019. }
  3020.  
  3021. void special_node::tprint_start(troff_output_file *out)
  3022. {
  3023.   out->start_special();
  3024. }
  3025.  
  3026. void special_node::tprint_char(troff_output_file *out, unsigned char c)
  3027. {
  3028.   out->special_char(c);
  3029. }
  3030.  
  3031. void special_node::tprint_end(troff_output_file *out)
  3032. {
  3033.   out->end_special();
  3034. }
  3035.  
  3036. /* composite_node */
  3037.  
  3038. class composite_node : public node {
  3039.   charinfo *ci;
  3040.   node *n;
  3041.   tfont *tf;
  3042. public:
  3043.   composite_node(node *, charinfo *, tfont *, node * = 0);
  3044.   ~composite_node();
  3045.   node *copy();
  3046.   hunits width();
  3047.   node *last_char_node();
  3048.   units size();
  3049.   void tprint(troff_output_file *);
  3050.   hyphenation_type get_hyphenation_type();
  3051.   int overlaps_horizontally();
  3052.   int overlaps_vertically();
  3053.   void ascii_print(ascii_output_file *);
  3054.   void asciify(macro *);
  3055.   hyphen_list *get_hyphen_list(hyphen_list *tail);
  3056.   node *add_self(node *, hyphen_list **);
  3057.   tfont *get_tfont();
  3058.   int same(node *);
  3059.   const char *type();
  3060.   void vertical_extent(vunits *, vunits *);
  3061.   vunits vertical_width();
  3062. };
  3063.  
  3064. composite_node::composite_node(node *p, charinfo *c, tfont *t, node *x)
  3065. : node(x), n(p), ci(c), tf(t)
  3066. {
  3067. }
  3068.  
  3069. composite_node::~composite_node()
  3070. {
  3071.   delete_node_list(n);
  3072. }
  3073.  
  3074. node *composite_node::copy()
  3075. {
  3076.   return new composite_node(copy_node_list(n), ci, tf);
  3077. }
  3078.  
  3079. hunits composite_node::width()
  3080. {
  3081.   hunits x;
  3082.   if (tf->get_constant_space(&x))
  3083.     return x;
  3084.   x = H0;
  3085.   for (node *tem = n; tem; tem = tem->next)
  3086.     x += tem->width();
  3087.   hunits offset;
  3088.   if (tf->get_bold(&offset))
  3089.     x += offset;
  3090.   x += tf->get_track_kern();
  3091.   return x;
  3092. }
  3093.  
  3094. node *composite_node::last_char_node()
  3095. {
  3096.   return this;
  3097. }
  3098.  
  3099. vunits composite_node::vertical_width()
  3100. {
  3101.   vunits v = V0;
  3102.   for (node *tem = n; tem; tem = tem->next)
  3103.     v += tem->vertical_width();
  3104.   return v;
  3105. }
  3106.  
  3107. units composite_node::size()
  3108. {
  3109.   return tf->get_size().to_units();
  3110. }
  3111.  
  3112. hyphenation_type composite_node::get_hyphenation_type()
  3113. {
  3114.   return HYPHEN_MIDDLE;
  3115. }
  3116.  
  3117. int composite_node::overlaps_horizontally()
  3118. {
  3119.   return ci->overlaps_horizontally();
  3120. }
  3121.  
  3122. int composite_node::overlaps_vertically()
  3123. {
  3124.   return ci->overlaps_vertically();
  3125. }
  3126.  
  3127. void composite_node::asciify(macro *m)
  3128. {
  3129.   unsigned char c = ci->get_ascii_code();
  3130.   if (c != 0) {
  3131.     m->append(c);
  3132.     delete this;
  3133.   }
  3134.   else
  3135.     m->append(this);
  3136. }
  3137.  
  3138. void composite_node::ascii_print(ascii_output_file *ascii)
  3139. {
  3140.   unsigned char c = ci->get_ascii_code();
  3141.   if (c != 0)
  3142.     ascii->outc(c);
  3143.   else
  3144.     ascii->outs(ci->nm.contents());
  3145.  
  3146. }
  3147.  
  3148. hyphen_list *composite_node::get_hyphen_list(hyphen_list *tail)
  3149. {
  3150.   return new hyphen_list(ci->get_hyphenation_code(), tail);
  3151.  
  3152. }
  3153.  
  3154. node *composite_node::add_self(node *nn, hyphen_list **p)
  3155. {
  3156.   assert(ci->get_hyphenation_code() == (*p)->hyphenation_code);
  3157.   next = nn;
  3158.   nn = this;
  3159.   if ((*p)->hyphen)
  3160.     nn = nn->add_discretionary_hyphen();
  3161.   hyphen_list *pp = *p;
  3162.   *p = (*p)->next;
  3163.   delete pp;
  3164.   return nn;
  3165. }
  3166.  
  3167. tfont *composite_node::get_tfont()
  3168. {
  3169.   return tf;
  3170. }
  3171.  
  3172. node *reverse_node_list(node *n)
  3173. {
  3174.   node *r = 0;
  3175.   while (n) {
  3176.     node *tem = n;
  3177.     n = n->next;
  3178.     tem->next = r;
  3179.     r = tem;
  3180.   }
  3181.   return r;
  3182. }
  3183.  
  3184. void composite_node::vertical_extent(vunits *min, vunits *max)
  3185. {
  3186.   n = reverse_node_list(n);
  3187.   node_list_vertical_extent(n, min, max);
  3188.   n = reverse_node_list(n);
  3189. }
  3190.  
  3191. word_space_node::word_space_node(hunits d, node *x) : space_node(d, x)
  3192. {
  3193. }
  3194.  
  3195. word_space_node::word_space_node(hunits d, int s, node *x)
  3196. : space_node(d, s, x)
  3197. {
  3198. }
  3199.  
  3200. node *word_space_node::copy()
  3201. {
  3202.   return new word_space_node(n, set);
  3203. }
  3204.  
  3205. void word_space_node::tprint(troff_output_file *out)
  3206. {
  3207.   out->word_marker();
  3208.   space_node::tprint(out);
  3209. }
  3210.  
  3211. unbreakable_space_node::unbreakable_space_node(hunits d, node *x)
  3212. : word_space_node(d, x)
  3213. {
  3214. }
  3215.  
  3216. unbreakable_space_node::unbreakable_space_node(hunits d, int s, node *x)
  3217. : word_space_node(d, s, x)
  3218. {
  3219. }
  3220.  
  3221. node *unbreakable_space_node::copy()
  3222. {
  3223.   return new unbreakable_space_node(n, set);
  3224. }
  3225.  
  3226. breakpoint *unbreakable_space_node::get_breakpoints(hunits, int,
  3227.                             breakpoint *rest, int)
  3228. {
  3229.   return rest;
  3230. }
  3231.  
  3232. int unbreakable_space_node::nbreaks()
  3233. {
  3234.   return 0;
  3235. }
  3236.  
  3237. void unbreakable_space_node::split(int, node **, node **)
  3238. {
  3239.   assert(0);
  3240. }
  3241.  
  3242. int unbreakable_space_node::merge_space(hunits)
  3243. {
  3244.   return 0;
  3245. }
  3246.  
  3247. hvpair::hvpair()
  3248. {
  3249. }
  3250.  
  3251. draw_node::draw_node(char c, hvpair *p, int np, font_size s)
  3252.      : code(c), npoints(np), sz(s)
  3253. {
  3254.   point = new hvpair[npoints];
  3255.   for (int i = 0; i < npoints; i++)
  3256.     point[i] = p[i];
  3257. }
  3258.  
  3259. int draw_node::same(node *n)
  3260. {
  3261.   draw_node *nd = (draw_node *)n;
  3262.   if (code != nd->code || npoints != nd->npoints || sz != nd->sz)
  3263.     return 0;
  3264.   for (int i = 0; i < npoints; i++)
  3265.     if (point[i].h != nd->point[i].h || point[i].v != nd->point[i].v)
  3266.       return 0;
  3267.   return 1;
  3268. }
  3269.  
  3270. const char *draw_node::type()
  3271. {
  3272.   return "draw_node";
  3273. }
  3274.  
  3275. draw_node::~draw_node()
  3276. {
  3277.   if (point)
  3278.     a_delete point;
  3279. }
  3280.  
  3281. hunits draw_node::width()
  3282. {
  3283.   hunits x = H0;
  3284.   for (int i = 0; i < npoints; i++)
  3285.     x += point[i].h;
  3286.   return x;
  3287. }
  3288.  
  3289. vunits draw_node::vertical_width()
  3290. {
  3291.   if (code == 'e')
  3292.     return V0;
  3293.   vunits x = V0;
  3294.   for (int i = 0; i < npoints; i++)
  3295.     x += point[i].v;
  3296.   return x;
  3297. }
  3298.  
  3299. node *draw_node::copy()
  3300. {
  3301.   return new draw_node(code, point, npoints, sz);
  3302. }
  3303.  
  3304. void draw_node::tprint(troff_output_file *out)
  3305. {
  3306.   out->draw(code, point, npoints, sz);
  3307. }
  3308.   
  3309. /* tprint methods */
  3310.  
  3311. void glyph_node::tprint(troff_output_file *out)
  3312. {
  3313.   tfont *ptf = tf->get_plain();
  3314.   if (ptf == tf)
  3315.     out->put_char_width(ci, ptf, width(), H0);
  3316.   else {
  3317.     hunits offset;
  3318.     int bold = tf->get_bold(&offset);
  3319.     hunits w = ptf->get_width(ci);
  3320.     hunits k = H0;
  3321.     hunits x;
  3322.     int cs = tf->get_constant_space(&x);
  3323.     if (cs) {
  3324.       x -= w;
  3325.       if (bold)
  3326.     x -= offset;
  3327.       hunits x2 = x/2;
  3328.       out->right(x2);
  3329.       k = x - x2;
  3330.     }
  3331.     else
  3332.       k = tf->get_track_kern();
  3333.     if (bold) {
  3334.       out->put_char(ci, ptf);
  3335.       out->right(offset);
  3336.     }
  3337.     out->put_char_width(ci, ptf, w, k);
  3338.   }
  3339. }
  3340.  
  3341. void glyph_node::zero_width_tprint(troff_output_file *out)
  3342. {
  3343.   tfont *ptf = tf->get_plain();
  3344.   hunits offset;
  3345.   int bold = tf->get_bold(&offset);
  3346.   hunits x;
  3347.   int cs = tf->get_constant_space(&x);
  3348.   if (cs) {
  3349.     x -= ptf->get_width(ci);
  3350.     if (bold)
  3351.       x -= offset;
  3352.     x = x/2;
  3353.     out->right(x);
  3354.   }
  3355.   out->put_char(ci, ptf);
  3356.   if (bold) {
  3357.     out->right(offset);
  3358.     out->put_char(ci, ptf);
  3359.     out->right(-offset);
  3360.   }
  3361.   if (cs)
  3362.     out->right(-x);
  3363. }
  3364.  
  3365. void break_char_node::tprint(troff_output_file *t)
  3366. {
  3367.   ch->tprint(t);
  3368. }
  3369.  
  3370. void break_char_node::zero_width_tprint(troff_output_file *t)
  3371. {
  3372.   ch->zero_width_tprint(t);
  3373. }
  3374.  
  3375. void hline_node::tprint(troff_output_file *out)
  3376. {
  3377.   if (x < H0) {
  3378.     out->right(x);
  3379.     x = -x;
  3380.   }
  3381.   if (n == 0) {
  3382.     out->right(x);
  3383.     return;
  3384.   }
  3385.   hunits w = n->width();
  3386.   if (w <= H0) {
  3387.     error("horizontal line drawing character must have positive width");
  3388.     out->right(x);
  3389.     return;
  3390.   }
  3391.   int i = int(x/w);
  3392.   if (i == 0) {
  3393.     hunits xx = x - w;
  3394.     hunits xx2 = xx/2;
  3395.     out->right(xx2);
  3396.     n->tprint(out);
  3397.     out->right(xx - xx2);
  3398.   }
  3399.   else {
  3400.     hunits rem = x - w*i;
  3401.     if (rem > H0)
  3402.       if (n->overlaps_horizontally()) {
  3403.     n->tprint(out);
  3404.     out->right(rem - w);
  3405.       }
  3406.       else
  3407.     out->right(rem);
  3408.     while (--i >= 0)
  3409.       n->tprint(out);
  3410.   }
  3411. }
  3412.  
  3413. void vline_node::tprint(troff_output_file *out)
  3414. {
  3415.   if (n == 0) {
  3416.     out->down(x);
  3417.     return;
  3418.   }
  3419.   vunits h = n->size();
  3420.   int overlaps = n->overlaps_vertically();
  3421.   vunits y = x;
  3422.   if (y < V0) {
  3423.     y = -y;
  3424.     int i = y / h;
  3425.     vunits rem = y - i*h;
  3426.     if (i == 0) {
  3427.       out->right(n->width());
  3428.       out->down(-rem);
  3429.     }
  3430.     else {
  3431.       while (--i > 0) {
  3432.     n->zero_width_tprint(out);
  3433.     out->down(-h);
  3434.       }
  3435.       if (overlaps) {
  3436.     n->zero_width_tprint(out);
  3437.     out->down(-rem);
  3438.     n->tprint(out);
  3439.     out->down(-h);
  3440.       }
  3441.       else {
  3442.     n->tprint(out);
  3443.     out->down(-h - rem);
  3444.       }
  3445.     }
  3446.   }
  3447.   else {
  3448.     int i = y / h;
  3449.     vunits rem = y - i*h;
  3450.     if (i == 0) {
  3451.       out->down(rem);
  3452.       out->right(n->width());
  3453.     }
  3454.     else {
  3455.       out->down(h);
  3456.       if (overlaps)
  3457.     n->zero_width_tprint(out);
  3458.       out->down(rem);
  3459.       while (--i > 0) {
  3460.     n->zero_width_tprint(out);
  3461.     out->down(h);
  3462.       }
  3463.       n->tprint(out);
  3464.     }
  3465.   }
  3466. }
  3467.  
  3468. void zero_width_node::tprint(troff_output_file *out)
  3469. {
  3470.   if (!n)
  3471.     return;
  3472.   if (!n->next) {
  3473.     n->zero_width_tprint(out);
  3474.     return;
  3475.   }
  3476.   int hpos = out->get_hpos();
  3477.   int vpos = out->get_vpos();
  3478.   node *tem = n;
  3479.   while (tem) {
  3480.     tem->tprint(out);
  3481.     tem = tem->next;
  3482.   }
  3483.   out->moveto(hpos, vpos);
  3484. }
  3485.  
  3486. void overstrike_node::tprint(troff_output_file *out)
  3487. {
  3488.   hunits pos = H0;
  3489.   for (node *tem = list; tem; tem = tem->next) {
  3490.     hunits x = (max_width - tem->width())/2;
  3491.     out->right(x - pos);
  3492.     pos = x;
  3493.     tem->zero_width_tprint(out);
  3494.   }
  3495.   out->right(max_width - pos);
  3496. }
  3497.  
  3498. void bracket_node::tprint(troff_output_file *out)
  3499. {
  3500.   if (list == 0)
  3501.     return;
  3502.   int npieces = 0;
  3503.   for (node *tem = list; tem; tem = tem->next)
  3504.     ++npieces;
  3505.   vunits h = list->size();
  3506.   vunits totalh = h*npieces;
  3507.   vunits y = (totalh - h)/2;
  3508.   out->down(y);
  3509.   for (tem = list; tem; tem = tem->next) {
  3510.     tem->zero_width_tprint(out);
  3511.     out->down(-h);
  3512.   }
  3513.   out->right(max_width);
  3514.   out->down(totalh - y);
  3515. }
  3516.  
  3517. void node::tprint(troff_output_file *)
  3518. {
  3519. }
  3520.  
  3521. void node::zero_width_tprint(troff_output_file *out)
  3522. {
  3523.   int hpos = out->get_hpos();
  3524.   int vpos = out->get_vpos();
  3525.   tprint(out);
  3526.   out->moveto(hpos, vpos);
  3527. }
  3528.  
  3529. void space_node::tprint(troff_output_file *out)
  3530. {
  3531.   out->right(n);
  3532. }
  3533.  
  3534. void hmotion_node::tprint(troff_output_file *out)
  3535. {
  3536.   out->right(n);
  3537. }
  3538.  
  3539. void vmotion_node::tprint(troff_output_file *out)
  3540. {
  3541.   out->down(n);
  3542. }
  3543.  
  3544. void kern_pair_node::tprint(troff_output_file *out)
  3545. {
  3546.   n1->tprint(out);
  3547.   out->right(amount);
  3548.   n2->tprint(out);
  3549. }
  3550.  
  3551. static void tprint_reverse_node_list(troff_output_file *out, node *n)
  3552. {
  3553.   if (n == 0)
  3554.     return;
  3555.   tprint_reverse_node_list(out, n->next);
  3556.   n->tprint(out);
  3557. }
  3558.  
  3559. void dbreak_node::tprint(troff_output_file *out)
  3560. {
  3561.   tprint_reverse_node_list(out, none);
  3562. }
  3563.  
  3564. void composite_node::tprint(troff_output_file *out)
  3565. {
  3566.   hunits bold_offset;
  3567.   int is_bold = tf->get_bold(&bold_offset);
  3568.   hunits track_kern = tf->get_track_kern();
  3569.   hunits constant_space;
  3570.   int is_constant_spaced = tf->get_constant_space(&constant_space);
  3571.   hunits x = H0;
  3572.   if (is_constant_spaced) {
  3573.     x = constant_space;
  3574.     for (node *tem = n; tem; tem = tem->next)
  3575.       x -= tem->width();
  3576.     if (is_bold)
  3577.       x -= bold_offset;
  3578.     hunits x2 = x/2;
  3579.     out->right(x2);
  3580.     x -= x2;
  3581.   }
  3582.   if (is_bold) {
  3583.     int hpos = out->get_hpos();
  3584.     int vpos = out->get_vpos();
  3585.     tprint_reverse_node_list(out, n);
  3586.     out->moveto(hpos, vpos);
  3587.     out->right(bold_offset);
  3588.   }
  3589.   tprint_reverse_node_list(out, n);
  3590.   if (is_constant_spaced)
  3591.     out->right(x);
  3592.   else
  3593.     out->right(track_kern);
  3594. }
  3595.  
  3596. node *make_composite_node(charinfo *s, environment *env)
  3597. {
  3598.   int fontno = env_definite_font(env);
  3599.   if (fontno < 0) {
  3600.     error("no current font");
  3601.     return 0;
  3602.   }
  3603.   assert(fontno < font_table_size && font_table[fontno] != 0);
  3604.   node *n = charinfo_to_node_list(s, env);
  3605.   font_size fs = env->get_font_size();
  3606.   int char_height = env->get_char_height();
  3607.   int char_slant = env->get_char_slant();
  3608.   tfont *tf = font_table[fontno]->get_tfont(fs, char_height, char_slant,
  3609.                         fontno);
  3610.   if (env->is_composite())
  3611.     tf = tf->get_plain();
  3612.   return new composite_node(n, s, tf);
  3613. }
  3614.  
  3615. node *make_glyph_node(charinfo *s, environment *env, int no_error_message = 0)
  3616. {
  3617.   int fontno = env_definite_font(env);
  3618.   if (fontno < 0) {
  3619.     error("no current font");
  3620.     return 0;
  3621.   }
  3622.   assert(fontno < font_table_size && font_table[fontno] != 0);
  3623.   int fn = fontno;
  3624.   int found = font_table[fontno]->contains(s);
  3625.   if (!found) {
  3626.     if (s->numbered()) {
  3627.       if (!no_error_message)
  3628.     warning(WARN_CHAR, "can't find numbered character %1",
  3629.         s->get_number());
  3630.       return 0;
  3631.     }
  3632.     special_font_list *sf = font_table[fontno]->sf;
  3633.     while (sf != 0 && !found) {
  3634.       fn = sf->n;
  3635.       if (font_table[fn])
  3636.     found = font_table[fn]->contains(s);
  3637.       sf = sf->next;
  3638.     }
  3639.     if (!found) {
  3640.       sf = global_special_fonts;
  3641.       while (sf != 0 && !found) {
  3642.     fn = sf->n;
  3643.     if (font_table[fn])
  3644.       found = font_table[fn]->contains(s);
  3645.     sf = sf->next;
  3646.       }
  3647.     }
  3648.     if (!found
  3649. #if 0    
  3650.     && global_special_fonts == 0 && font_table[fontno]->sf == 0
  3651. #endif
  3652.     ) {
  3653.       for (fn = 0; fn < font_table_size; fn++)
  3654.     if (font_table[fn] 
  3655.         && font_table[fn]->is_special()
  3656.         && font_table[fn]->contains(s)) {
  3657.           found = 1;
  3658.           break;
  3659.         }
  3660.     }
  3661.     if (!found) {
  3662.       if (!no_error_message && s->first_time_not_found()) {
  3663.     unsigned char input_code = s->get_ascii_code();
  3664.     if (input_code != 0) {
  3665.       if (csgraph(input_code))
  3666.         warning(WARN_CHAR, "can't find character `%1'", input_code);
  3667.       else
  3668.         warning(WARN_CHAR, "can't find character with input code %1",
  3669.             int(input_code));
  3670.     }
  3671.     else
  3672.       warning(WARN_CHAR, "can't find special character `%1'",
  3673.           s->nm.contents());
  3674.       }
  3675.       return 0;
  3676.     }
  3677.   }
  3678.   font_size fs = env->get_font_size();
  3679.   int char_height = env->get_char_height();
  3680.   int char_slant = env->get_char_slant();
  3681.   tfont *tf = font_table[fontno]->get_tfont(fs, char_height, char_slant, fn);
  3682.   if (env->is_composite())
  3683.     tf = tf->get_plain();
  3684.   return new glyph_node(s, tf);
  3685. }
  3686.  
  3687. node *make_node(charinfo *ci, environment *env)
  3688. {
  3689.   switch (ci->get_special_translation()) {
  3690.   case charinfo::TRANSLATE_SPACE:
  3691.     return new space_char_hmotion_node(env->get_space_width());
  3692.   case charinfo::TRANSLATE_DUMMY:
  3693.     return new dummy_node;
  3694.   case charinfo::TRANSLATE_HYPHEN_INDICATOR:
  3695.     error("translation to \\% ignored in this context");
  3696.     break;
  3697.   }
  3698.   charinfo *tem = ci->get_translation();
  3699.   if (tem)
  3700.     ci = tem;
  3701.   macro *mac = ci->get_macro();
  3702.   if (mac)
  3703.     return make_composite_node(ci, env);
  3704.   else
  3705.     return make_glyph_node(ci, env);
  3706. }
  3707.  
  3708. int character_exists(charinfo *ci, environment *env)
  3709. {
  3710.   if (ci->get_special_translation() != charinfo::TRANSLATE_NONE)
  3711.     return 1;
  3712.   charinfo *tem = ci->get_translation();
  3713.   if (tem)
  3714.     ci = tem;
  3715.   if (ci->get_macro())
  3716.     return 1;
  3717.   node *nd = make_glyph_node(ci, env, 1);
  3718.   if (nd) {
  3719.     delete nd;
  3720.     return 1;
  3721.   }
  3722.   return 0;
  3723. }
  3724.  
  3725. node *node::add_char(charinfo *ci, environment *env, hunits *widthp)
  3726. {
  3727.   node *res;
  3728.   switch (ci->get_special_translation()) {
  3729.   case charinfo::TRANSLATE_SPACE:
  3730.     res = new space_char_hmotion_node(env->get_space_width(), this);
  3731.     *widthp += res->width();
  3732.     return res;
  3733.   case charinfo::TRANSLATE_DUMMY:
  3734.     return new dummy_node(this);
  3735.   case charinfo::TRANSLATE_HYPHEN_INDICATOR:
  3736.     return add_discretionary_hyphen();
  3737.   }
  3738.   charinfo *tem = ci->get_translation();
  3739.   if (tem)
  3740.     ci = tem;
  3741.   macro *mac = ci->get_macro();
  3742.   if (mac) {
  3743.     res = make_composite_node(ci, env);
  3744.     if (res) {
  3745.       res->next = this;
  3746.       *widthp += res->width();
  3747.     }
  3748.     else
  3749.       return this;
  3750.   }
  3751.   else {
  3752.     node *gn = make_glyph_node(ci, env);
  3753.     if (gn == 0)
  3754.       return this;
  3755.     else {
  3756.       hunits old_width = width();
  3757.       node *p = gn->merge_self(this);
  3758.       if (p == 0) {
  3759.     *widthp += gn->width();
  3760.     gn->next = this;
  3761.     res = gn;
  3762.       }
  3763.       else {
  3764.     *widthp += p->width() - old_width;
  3765.     res = p;
  3766.       }
  3767.     }
  3768.   }
  3769.   int break_code = 0;
  3770.   if (ci->can_break_before())
  3771.     break_code = 1;
  3772.   if (ci->can_break_after())
  3773.     break_code |= 2;
  3774.   if (break_code) {
  3775.     node *next1 = res->next;
  3776.     res->next = 0;
  3777.     res = new break_char_node(res, break_code, next1);
  3778.   }
  3779.   return res;
  3780. }
  3781.  
  3782.  
  3783. #ifdef __GNUG__
  3784. inline
  3785. #endif
  3786. int same_node(node *n1, node *n2)
  3787. {
  3788.   if (n1 != 0) {
  3789.     if (n2 != 0)
  3790.       return n1->type() == n2->type() && n1->same(n2);
  3791.     else
  3792.       return 0;
  3793.   }
  3794.   else
  3795.     return n2 == 0;
  3796. }
  3797.  
  3798. int same_node_list(node *n1, node *n2)
  3799. {
  3800.   while (n1 && n2) {
  3801.     if (n1->type() != n2->type() || !n1->same(n2))
  3802.       return 0;
  3803.     n1 = n1->next;
  3804.     n2 = n2->next;
  3805.   }
  3806.   return !n1 && !n2;
  3807. }
  3808.  
  3809. int extra_size_node::same(node *nd)
  3810. {
  3811.   return n == ((extra_size_node *)nd)->n;
  3812. }
  3813.  
  3814. const char *extra_size_node::type()
  3815. {
  3816.   return "extra_size_node";
  3817. }
  3818.  
  3819. int vertical_size_node::same(node *nd)
  3820. {
  3821.   return n == ((vertical_size_node *)nd)->n;
  3822. }
  3823.  
  3824. const char *vertical_size_node::type()
  3825. {
  3826.   return "vertical_size_node";
  3827. }
  3828.  
  3829. int hmotion_node::same(node *nd)
  3830. {
  3831.   return n == ((hmotion_node *)nd)->n;
  3832. }
  3833.  
  3834. const char *hmotion_node::type()
  3835. {
  3836.   return "hmotion_node";
  3837. }
  3838.  
  3839. int space_char_hmotion_node::same(node *nd)
  3840. {
  3841.   return n == ((space_char_hmotion_node *)nd)->n;
  3842. }
  3843.  
  3844. const char *space_char_hmotion_node::type()
  3845. {
  3846.   return "space_char_hmotion_node";
  3847. }
  3848.  
  3849. int vmotion_node::same(node *nd)
  3850. {
  3851.   return n == ((vmotion_node *)nd)->n;
  3852. }
  3853.  
  3854. const char *vmotion_node::type()
  3855. {
  3856.   return "vmotion_node";
  3857. }
  3858.  
  3859. int hline_node::same(node *nd)
  3860. {
  3861.   return x == ((hline_node *)nd)->x && same_node(n, ((hline_node *)nd)->n);
  3862. }
  3863.  
  3864. const char *hline_node::type()
  3865. {
  3866.   return "hline_node";
  3867. }
  3868.  
  3869. int vline_node::same(node *nd)
  3870. {
  3871.   return x == ((vline_node *)nd)->x && same_node(n, ((vline_node *)nd)->n);
  3872. }
  3873.  
  3874. const char *vline_node::type()
  3875. {
  3876.   return "vline_node";
  3877. }
  3878.  
  3879. int dummy_node::same(node * /*nd*/)
  3880. {
  3881.   return 1;
  3882. }
  3883.  
  3884. const char *dummy_node::type()
  3885. {
  3886.   return "dummy_node";
  3887. }
  3888.  
  3889. int transparent_dummy_node::same(node * /*nd*/)
  3890. {
  3891.   return 1;
  3892. }
  3893.  
  3894. const char *transparent_dummy_node::type()
  3895. {
  3896.   return "transparent_dummy_node";
  3897. }
  3898.  
  3899. int transparent_dummy_node::ends_sentence()
  3900. {
  3901.   return 2;
  3902. }
  3903.  
  3904. int zero_width_node::same(node *nd)
  3905. {
  3906.   return same_node_list(n, ((zero_width_node *)nd)->n);
  3907. }
  3908.  
  3909. const char *zero_width_node::type()
  3910. {
  3911.   return "zero_width_node";
  3912. }
  3913.  
  3914. int italic_corrected_node::same(node *nd)
  3915. {
  3916.   return (x == ((italic_corrected_node *)nd)->x
  3917.       && same_node(n, ((italic_corrected_node *)nd)->n));
  3918. }
  3919.  
  3920. const char *italic_corrected_node::type()
  3921. {
  3922.   return "italic_corrected_node";
  3923. }
  3924.  
  3925.  
  3926. left_italic_corrected_node::left_italic_corrected_node(node *x)
  3927. : n(0), node(x)
  3928. {
  3929. }
  3930.  
  3931. left_italic_corrected_node::~left_italic_corrected_node()
  3932. {
  3933.   delete n;
  3934. }
  3935.  
  3936. node *left_italic_corrected_node::merge_glyph_node(glyph_node *gn)
  3937. {
  3938.   if (n == 0) {
  3939.     hunits lic = gn->left_italic_correction();
  3940.     if (!lic.is_zero()) {
  3941.       x = lic;
  3942.       n = gn;
  3943.       return this;
  3944.     }
  3945.   }
  3946.   else {
  3947.     node *nd = n->merge_glyph_node(gn);
  3948.     if (nd) {
  3949.       n = nd;
  3950.       x = n->left_italic_correction();
  3951.       return this;
  3952.     }
  3953.   }
  3954.   return 0;
  3955. }
  3956.  
  3957. node *left_italic_corrected_node::copy()
  3958. {
  3959.   left_italic_corrected_node *nd = new left_italic_corrected_node;
  3960.   if (n) {
  3961.     nd->n = n->copy();
  3962.     nd->x = x;
  3963.   }
  3964.   return nd;
  3965. }
  3966.  
  3967. void left_italic_corrected_node::tprint(troff_output_file *out)
  3968. {
  3969.   if (n) {
  3970.     out->right(x);
  3971.     n->tprint(out);
  3972.   }
  3973. }
  3974.  
  3975. const char *left_italic_corrected_node::type()
  3976. {
  3977.   return "left_italic_corrected_node";
  3978. }
  3979.  
  3980. int left_italic_corrected_node::same(node *nd)
  3981. {
  3982.   return (x == ((left_italic_corrected_node *)nd)->x
  3983.       && same_node(n, ((left_italic_corrected_node *)nd)->n));
  3984. }
  3985.  
  3986. void left_italic_corrected_node::ascii_print(ascii_output_file *out)
  3987. {
  3988.   if (n)
  3989.     n->ascii_print(out);
  3990. }
  3991.  
  3992. hunits left_italic_corrected_node::width()
  3993. {
  3994.   return n ? n->width() + x : H0;
  3995. }
  3996.  
  3997. void left_italic_corrected_node::vertical_extent(vunits *min, vunits *max)
  3998. {
  3999.   if (n)
  4000.     n->vertical_extent(min, max);
  4001.   else
  4002.     node::vertical_extent(min, max);
  4003. }
  4004.  
  4005. hunits left_italic_corrected_node::skew()
  4006. {
  4007.   return n ? n->skew() + x/2 : H0;
  4008. }
  4009.  
  4010. hunits left_italic_corrected_node::subscript_correction()
  4011. {
  4012.   return n ? n->subscript_correction() : H0;
  4013. }
  4014.  
  4015. hunits left_italic_corrected_node::italic_correction()
  4016. {
  4017.   return n ? n->italic_correction() : H0;
  4018. }
  4019.  
  4020. int left_italic_corrected_node::ends_sentence()
  4021. {
  4022.   return n ? n->ends_sentence() : 0;
  4023. }
  4024.  
  4025. int left_italic_corrected_node::overlaps_horizontally()
  4026. {
  4027.   return n ? n->overlaps_horizontally() : 0;
  4028. }
  4029.  
  4030. int left_italic_corrected_node::overlaps_vertically()
  4031. {
  4032.   return n ? n->overlaps_vertically() : 0;
  4033. }
  4034.  
  4035. node *left_italic_corrected_node::last_char_node()
  4036. {
  4037.   return n ? n->last_char_node() : 0;
  4038. }
  4039.  
  4040. tfont *left_italic_corrected_node::get_tfont()
  4041. {
  4042.   return n ? n->get_tfont() : 0;
  4043. }
  4044.  
  4045. hyphenation_type left_italic_corrected_node::get_hyphenation_type()
  4046. {
  4047.   if (n)
  4048.     return n->get_hyphenation_type();
  4049.   else
  4050.     return HYPHEN_MIDDLE;
  4051. }
  4052.  
  4053. hyphen_list *left_italic_corrected_node::get_hyphen_list(hyphen_list *tail)
  4054. {
  4055.   return n ? n->get_hyphen_list(tail) : tail;
  4056. }
  4057.  
  4058. node *left_italic_corrected_node::add_self(node *nd, hyphen_list **p)
  4059. {
  4060.   if (n) {
  4061.     nd = new left_italic_corrected_node(nd);
  4062.     nd = n->add_self(nd, p);
  4063.     n = 0;
  4064.     delete this;
  4065.   }
  4066.   return nd;
  4067. }
  4068.  
  4069. int left_italic_corrected_node::character_type()
  4070. {
  4071.   return n ? n->character_type() : 0;
  4072. }
  4073.  
  4074. int overstrike_node::same(node *nd)
  4075. {
  4076.   return same_node_list(list, ((overstrike_node *)nd)->list);
  4077. }
  4078.  
  4079. const char *overstrike_node::type()
  4080. {
  4081.   return "overstrike_node";
  4082. }
  4083.  
  4084. int bracket_node::same(node *nd)
  4085. {
  4086.   return same_node_list(list, ((bracket_node *)nd)->list);
  4087. }
  4088.  
  4089. const char *bracket_node::type()
  4090. {
  4091.   return "bracket_node";
  4092. }
  4093.  
  4094. int composite_node::same(node *nd)
  4095. {
  4096.   return ci == ((composite_node *)nd)->ci
  4097.     && same_node_list(n, ((composite_node *)nd)->n);
  4098. }
  4099.  
  4100. const char *composite_node::type()
  4101. {
  4102.   return "composite_node";
  4103. }
  4104.  
  4105. int glyph_node::same(node *nd)
  4106. {
  4107.   return ci == ((glyph_node *)nd)->ci && tf == ((glyph_node *)nd)->tf;
  4108. }
  4109.  
  4110. const char *glyph_node::type()
  4111. {
  4112.   return "glyph_node";
  4113. }
  4114.  
  4115. int ligature_node::same(node *nd)
  4116. {
  4117.   return (same_node(n1, ((ligature_node *)nd)->n1) 
  4118.       && same_node(n2, ((ligature_node *)nd)->n2)
  4119.       && glyph_node::same(nd));
  4120. }
  4121.  
  4122. const char *ligature_node::type()
  4123. {
  4124.   return "ligature_node";
  4125. }
  4126.  
  4127. int kern_pair_node::same(node *nd)
  4128. {
  4129.   return (amount == ((kern_pair_node *)nd)->amount
  4130.       && same_node(n1, ((kern_pair_node *)nd)->n1)
  4131.       && same_node(n2, ((kern_pair_node *)nd)->n2));
  4132. }
  4133.  
  4134. const char *kern_pair_node::type()
  4135. {
  4136.   return "kern_pair_node";
  4137. }
  4138.  
  4139. int dbreak_node::same(node *nd)
  4140. {
  4141.   return (same_node_list(none, ((dbreak_node *)nd)->none)
  4142.       && same_node_list(pre, ((dbreak_node *)nd)->pre)
  4143.       && same_node_list(post, ((dbreak_node *)nd)->post));
  4144. }
  4145.  
  4146. const char *dbreak_node::type()
  4147. {
  4148.   return "dbreak_node";
  4149. }
  4150.  
  4151. int break_char_node::same(node *nd)
  4152. {
  4153.   return (break_code == ((break_char_node *)nd)->break_code
  4154.       && same_node(ch, ((break_char_node *)nd)->ch));
  4155. }
  4156.  
  4157. const char *break_char_node::type()
  4158. {
  4159.   return "break_char_node";
  4160. }
  4161.  
  4162. int line_start_node::same(node * /*nd*/)
  4163. {
  4164.   return 1;
  4165. }
  4166.  
  4167. const char *line_start_node::type()
  4168. {
  4169.   return "line_start_node";
  4170. }
  4171.  
  4172. int space_node::same(node *nd)
  4173. {
  4174.   return n == ((space_node *)nd)->n && set == ((space_node *)nd)->set;
  4175. }
  4176.  
  4177. const char *space_node::type()
  4178. {
  4179.   return "space_node";
  4180. }
  4181.  
  4182. int word_space_node::same(node *nd)
  4183. {
  4184.   return (n == ((word_space_node *)nd)->n
  4185.       && set == ((word_space_node *)nd)->set);
  4186. }
  4187.  
  4188. const char *word_space_node::type()
  4189. {
  4190.   return "word_space_node";
  4191. }
  4192.  
  4193. int unbreakable_space_node::same(node *nd)
  4194. {
  4195.   return (n == ((unbreakable_space_node *)nd)->n
  4196.       && set == ((unbreakable_space_node *)nd)->set);
  4197. }
  4198.  
  4199. const char *unbreakable_space_node::type()
  4200. {
  4201.   return "unbreakable_space_node";
  4202. }
  4203.  
  4204. int diverted_space_node::same(node *nd)
  4205. {
  4206.   return n == ((diverted_space_node *)nd)->n;
  4207. }
  4208.  
  4209. const char *diverted_space_node::type()
  4210. {
  4211.   return "diverted_space_node";
  4212. }
  4213.  
  4214. int diverted_copy_file_node::same(node *nd)
  4215. {
  4216.   return filename == ((diverted_copy_file_node *)nd)->filename;
  4217. }
  4218.  
  4219. const char *diverted_copy_file_node::type()
  4220. {
  4221.   return "diverted_copy_file_node";
  4222. }
  4223.  
  4224. // Grow the font_table so that its size is > n.
  4225.  
  4226. static void grow_font_table(int n)
  4227. {
  4228.   assert(n >= font_table_size);
  4229.   font_info **old_font_table = font_table;
  4230.   int old_font_table_size = font_table_size;
  4231.   font_table_size = font_table_size ? (font_table_size*3)/2 : 10;
  4232.   if (font_table_size <= n)
  4233.     font_table_size = n + 10;
  4234.   font_table = new font_info *[font_table_size];
  4235.   if (old_font_table_size)
  4236.     memcpy(font_table, old_font_table,
  4237.        old_font_table_size*sizeof(font_info *));
  4238.   a_delete old_font_table;
  4239.   for (int i = old_font_table_size; i < font_table_size; i++)
  4240.     font_table[i] = 0;
  4241. }
  4242.  
  4243. dictionary font_translation_dictionary(17);
  4244.  
  4245. static symbol get_font_translation(symbol nm)
  4246. {
  4247.   void *p = font_translation_dictionary.lookup(nm);
  4248.   return p ? symbol((char *)p) : nm;
  4249. }
  4250.  
  4251. dictionary font_dictionary(50);
  4252.  
  4253. static int mount_font_no_translate(int n, symbol name, symbol external_name)
  4254. {
  4255.   assert(n >= 0);
  4256.   // We store the address of this char in font_dictionary to indicate
  4257.   // that we've previously tried to mount the font and failed.
  4258.   static char a_char;
  4259.   font *fm = 0;
  4260.   void *p = font_dictionary.lookup(external_name);
  4261.   if (p == 0) {
  4262.     int not_found;
  4263.     fm = font::load_font(external_name.contents(), ¬_found);
  4264.     if (!fm) {
  4265.       if (not_found)
  4266.     warning(WARN_FONT, "can't find font `%1'", external_name.contents());
  4267.       font_dictionary.lookup(external_name, &a_char);
  4268.       return 0;
  4269.     }
  4270.     font_dictionary.lookup(name, fm);
  4271.   }
  4272.   else if (p == &a_char) {
  4273. #if 0
  4274.     error("invalid font `%1'", external_name.contents());
  4275. #endif
  4276.     return 0;
  4277.   }
  4278.   else
  4279.     fm = (font*)p;
  4280.   if (n >= font_table_size) {
  4281.     if (n - font_table_size > 1000) {
  4282.       error("font position too much larger than first unused position");
  4283.       return 0;
  4284.     }
  4285.     grow_font_table(n);
  4286.   }
  4287.   else if (font_table[n] != 0)
  4288.     delete font_table[n];
  4289.   font_table[n] = new font_info(name, n, external_name, fm);
  4290.   invalidate_fontno(n);
  4291.   return 1;
  4292. }
  4293.  
  4294. int mount_font(int n, symbol name, symbol external_name)
  4295. {
  4296.   assert(n >= 0);
  4297.   name = get_font_translation(name);
  4298.   if (external_name.is_null())
  4299.     external_name = name;
  4300.   else
  4301.     external_name = get_font_translation(external_name);
  4302.   return mount_font_no_translate(n, name, external_name);
  4303. }
  4304.  
  4305. void mount_style(int n, symbol name)
  4306. {
  4307.   assert(n >= 0);
  4308.   if (n >= font_table_size) {
  4309.     if (n - font_table_size > 1000) {
  4310.       error("font position too much larger than first unused position");
  4311.       return;
  4312.     }
  4313.     grow_font_table(n);
  4314.   }
  4315.   else if (font_table[n] != 0)
  4316.     delete font_table[n];
  4317.   font_table[n] = new font_info(get_font_translation(name), n, NULL_SYMBOL, 0);
  4318.   invalidate_fontno(n);
  4319. }
  4320.  
  4321. /* global functions */
  4322.  
  4323. void font_translate()
  4324. {
  4325.   symbol from = get_name(1);
  4326.   if (!from.is_null()) {
  4327.     symbol to = get_name();
  4328.     if (to.is_null() || from == to)
  4329.       font_translation_dictionary.remove(from);
  4330.     else
  4331.       font_translation_dictionary.lookup(from, (void *)to.contents());
  4332.   }
  4333.   skip_line();
  4334. }
  4335.  
  4336. void font_position()
  4337. {
  4338.   int n;
  4339.   if (get_integer(&n)) {
  4340.     if (n < 0)
  4341.       error("negative font position");
  4342.     else {
  4343.       symbol internal_name = get_name(1);
  4344.       if (!internal_name.is_null()) {
  4345.     symbol external_name = get_long_name(0);
  4346.     mount_font(n, internal_name, external_name); // ignore error
  4347.       }
  4348.     }
  4349.   }
  4350.   skip_line();
  4351. }
  4352.  
  4353. font_family::font_family(symbol s)
  4354. : nm(s), map_size(10)
  4355. {
  4356.   map = new int[map_size];
  4357.   for (int i = 0; i < map_size; i++)
  4358.     map[i] = -1;
  4359. }
  4360.  
  4361. font_family::~font_family()
  4362. {
  4363.   a_delete map;
  4364. }
  4365.  
  4366. int font_family::make_definite(int i)
  4367. {
  4368.   if (i >= 0) {
  4369.     if (i < map_size && map[i] >= 0)
  4370.       return map[i];
  4371.     else {
  4372.       if (i < font_table_size && font_table[i] != 0) {
  4373.     if (i >= map_size) {
  4374.       int old_map_size = map_size;
  4375.       int *old_map = map;
  4376.       map_size *= 3;
  4377.       map_size /= 2;
  4378.       if (i >= map_size)
  4379.         map_size = i + 10;
  4380.       map = new int[map_size];
  4381.       memcpy(map, old_map, old_map_size*sizeof(int));
  4382.       a_delete old_map;
  4383.       for (int j = old_map_size; j < map_size; j++)
  4384.         map[j] = -1;
  4385.     }
  4386.     if (font_table[i]->is_style()) {
  4387.       symbol sty = font_table[i]->get_name();
  4388.       symbol f = concat(nm, sty);
  4389.       int n;
  4390.       // don't use symbol_fontno, because that might return a style
  4391.       // and because we don't want to translate the name
  4392.       for (n = 0; n < font_table_size; n++)
  4393.         if (font_table[n] != 0 && font_table[n]->is_named(f)
  4394.         && !font_table[n]->is_style())
  4395.           break;
  4396.       if (n >= font_table_size) {
  4397.         n = next_available_font_position();
  4398.         if (!mount_font_no_translate(n, f, f))
  4399.           return -1;
  4400.       }
  4401.       return map[i] = n;
  4402.     }
  4403.     else
  4404.       return map[i] = i;
  4405.       }
  4406.       else
  4407.     return -1;
  4408.     }
  4409.   }
  4410.   else
  4411.     return -1;
  4412. }
  4413.  
  4414. dictionary family_dictionary(5);
  4415.  
  4416. font_family *lookup_family(symbol nm)
  4417. {
  4418.   font_family *f = (font_family *)family_dictionary.lookup(nm);
  4419.   if (!f) {
  4420.     f = new font_family(nm);
  4421.     (void)family_dictionary.lookup(nm, f);
  4422.   }
  4423.   return f;
  4424.  
  4425. static void invalidate_fontno(int n)
  4426. {
  4427.   assert(n >= 0 && n < font_table_size);
  4428.   dictionary_iterator iter(family_dictionary);
  4429.   symbol nm;
  4430.   font_family *fam;
  4431.   while (iter.get(&nm, (void **)&fam)) {
  4432.     int map_size = fam->map_size;
  4433.     if (n < map_size)
  4434.       fam->map[n] = -1;
  4435.     for (int i = 0; i < map_size; i++)
  4436.       if (fam->map[i] == n)
  4437.     fam->map[i] = -1;
  4438.   }
  4439. }
  4440.  
  4441. void style()
  4442. {
  4443.   int n;
  4444.   if (get_integer(&n)) {
  4445.     if (n < 0)
  4446.       error("negative font position");
  4447.     else {
  4448.       symbol internal_name = get_name(1);
  4449.       if (!internal_name.is_null())
  4450.     mount_style(n, internal_name);
  4451.     }
  4452.   }
  4453.   skip_line();
  4454. }
  4455.  
  4456. static int get_fontno()
  4457. {
  4458.   int n;
  4459.   tok.skip();
  4460.   if (tok.delimiter()) {
  4461.     symbol s = get_name(1);
  4462.     if (!s.is_null()) {
  4463.       n = symbol_fontno(s);
  4464.       if (n < 0) {
  4465.     n = next_available_font_position();
  4466.     if (!mount_font(n, s))
  4467.       return -1;
  4468.       }
  4469.       return curenv->get_family()->make_definite(n);
  4470.     }
  4471.   }
  4472.   else if (get_integer(&n)) {
  4473.     if (n < 0 || n >= font_table_size || font_table[n] == 0)
  4474.       error("bad font number");
  4475.     else
  4476.       return curenv->get_family()->make_definite(n);
  4477.   }
  4478.   return -1;
  4479. }
  4480.  
  4481. static int underline_fontno = 2;
  4482.  
  4483. void underline_font()
  4484. {
  4485.   int n = get_fontno();
  4486.   if (n >= 0)
  4487.     underline_fontno = n;
  4488.   skip_line();
  4489. }
  4490.  
  4491. int get_underline_fontno()
  4492. {
  4493.   return underline_fontno;
  4494. }
  4495.     
  4496. static void read_special_fonts(special_font_list **sp)
  4497. {
  4498.   special_font_list *s = *sp;
  4499.   *sp = 0;
  4500.   while (s != 0) {
  4501.     special_font_list *tem = s;
  4502.     s = s->next;
  4503.     delete tem;
  4504.   }
  4505.   special_font_list **p = sp;
  4506.   while (has_arg()) {
  4507.     int i = get_fontno();
  4508.     if (i >= 0) {
  4509.       special_font_list *tem = new special_font_list;
  4510.       tem->n = i;
  4511.       tem->next = 0;
  4512.       *p = tem;
  4513.       p = &(tem->next);
  4514.     }
  4515.   }
  4516. }
  4517.  
  4518. void font_special_request()
  4519. {
  4520.   int n = get_fontno();
  4521.   if (n >= 0)
  4522.     read_special_fonts(&font_table[n]->sf); 
  4523.   skip_line();
  4524. }
  4525.  
  4526.   
  4527. void special_request()
  4528. {
  4529.   read_special_fonts(&global_special_fonts);
  4530.   skip_line();
  4531. }
  4532.  
  4533. int next_available_font_position()
  4534. {
  4535.   for (int i = 1; i < font_table_size && font_table[i] != 0; i++)
  4536.     ;
  4537.   return i;
  4538. }
  4539.  
  4540. int symbol_fontno(symbol s)
  4541. {
  4542.   s = get_font_translation(s);
  4543.   for (int i = 0; i < font_table_size; i++)
  4544.     if (font_table[i] != 0 && font_table[i]->is_named(s))
  4545.       return i;
  4546.   return -1;
  4547. }
  4548.  
  4549. int is_good_fontno(int n)
  4550. {
  4551.   return n >= 0 && n < font_table_size && font_table[n] != NULL;
  4552. }
  4553.  
  4554. int get_bold_fontno(int n)
  4555. {
  4556.   if (n >= 0 && n < font_table_size && font_table[n] != 0) {
  4557.     hunits offset;
  4558.     if (font_table[n]->get_bold(&offset))
  4559.       return offset.to_units() + 1;
  4560.     else
  4561.       return 0;
  4562.   }
  4563.   else
  4564.     return 0;
  4565. }
  4566.  
  4567. hunits env_digit_width(environment *env)
  4568. {
  4569.   node *n = make_glyph_node(charset_table['0'], env);
  4570.   if (n) {
  4571.     hunits x = n->width();
  4572.     delete n;
  4573.     return x;
  4574.   }
  4575.   else
  4576.     return H0;
  4577. }
  4578.  
  4579. hunits env_space_width(environment *env)
  4580. {
  4581.   int fn = env_definite_font(env);
  4582.   font_size fs = env->get_font_size();
  4583.   if (fn < 0 || fn >= font_table_size || font_table[fn] == 0)
  4584.     return scale(fs.to_units()/3, env->get_space_size(), 12);
  4585.   else
  4586.     return font_table[fn]->get_space_width(fs, env->get_space_size());
  4587. }
  4588.  
  4589. hunits env_sentence_space_width(environment *env)
  4590. {
  4591.   int fn = env_definite_font(env);
  4592.   font_size fs = env->get_font_size();
  4593.   if (fn < 0 || fn >= font_table_size || font_table[fn] == 0)
  4594.     return scale(fs.to_units()/3, env->get_sentence_space_size(), 12);
  4595.   else
  4596.     return font_table[fn]->get_space_width(fs, env->get_sentence_space_size());
  4597. }
  4598.  
  4599. hunits env_half_narrow_space_width(environment *env)
  4600. {
  4601.   int fn = env_definite_font(env);
  4602.   font_size fs = env->get_font_size();
  4603.   if (fn < 0 || fn >= font_table_size || font_table[fn] == 0)
  4604.     return 0;
  4605.   else
  4606.     return font_table[fn]->get_half_narrow_space_width(fs);
  4607. }
  4608.  
  4609. hunits env_narrow_space_width(environment *env)
  4610. {
  4611.   int fn = env_definite_font(env);
  4612.   font_size fs = env->get_font_size();
  4613.   if (fn < 0 || fn >= font_table_size || font_table[fn] == 0)
  4614.     return 0;
  4615.   else
  4616.     return font_table[fn]->get_narrow_space_width(fs);
  4617. }
  4618.  
  4619. void bold_font()
  4620. {
  4621.   int n = get_fontno();
  4622.   if (n >= 0) {
  4623.     if (has_arg()) {
  4624.       if (tok.delimiter()) {
  4625.     int f = get_fontno();
  4626.     if (f >= 0) {
  4627.       units offset;
  4628.       if (has_arg() && get_number(&offset, 'u') && offset >= 1)
  4629.         font_table[f]->set_conditional_bold(n, hunits(offset - 1));
  4630.       else
  4631.         font_table[f]->conditional_unbold(n);
  4632.     }
  4633.       }
  4634.       else {
  4635.     units offset;
  4636.     if (get_number(&offset, 'u') && offset >= 1)
  4637.       font_table[n]->set_bold(hunits(offset - 1));
  4638.     else
  4639.       font_table[n]->unbold();
  4640.       }
  4641.     }
  4642.     else
  4643.       font_table[n]->unbold();
  4644.   }
  4645.   skip_line();
  4646. }
  4647.  
  4648. track_kerning_function::track_kerning_function() : non_zero(0)
  4649. {
  4650. }
  4651.  
  4652. track_kerning_function::track_kerning_function(int min_s, hunits min_a, 
  4653.                            int max_s, hunits max_a)
  4654.      : non_zero(1), 
  4655.      min_size(min_s), min_amount(min_a),
  4656.      max_size(max_s), max_amount(max_a)
  4657. {
  4658. }
  4659.  
  4660. int track_kerning_function::operator==(const track_kerning_function &tk)
  4661. {
  4662.   if (non_zero)
  4663.     return (tk.non_zero
  4664.         && min_size == tk.min_size
  4665.         && min_amount == tk.min_amount
  4666.         && max_size == tk.max_size
  4667.         && max_amount == tk.max_amount);
  4668.   else
  4669.     return !tk.non_zero;
  4670. }
  4671.  
  4672. int track_kerning_function::operator!=(const track_kerning_function &tk)
  4673. {
  4674.   if (non_zero)
  4675.     return (!tk.non_zero
  4676.         || min_size != tk.min_size
  4677.         || min_amount != tk.min_amount
  4678.         || max_size != tk.max_size
  4679.         || max_amount != tk.max_amount);
  4680.   else
  4681.     return tk.non_zero;
  4682. }
  4683.  
  4684. hunits track_kerning_function::compute(int size)
  4685. {
  4686.   if (non_zero) {
  4687.     if (max_size <= min_size)
  4688.       return min_amount;
  4689.     else if (size <= min_size)
  4690.       return min_amount;
  4691.     else if (size >= max_size)
  4692.       return max_amount;
  4693.     else
  4694.       return (scale(max_amount, size - min_size, max_size - min_size)
  4695.           + scale(min_amount, max_size - size, max_size - min_size));
  4696.   }
  4697.   else
  4698.     return H0;
  4699. }
  4700.  
  4701. void track_kern()
  4702. {
  4703.   int n = get_fontno();
  4704.   if (n >= 0) {
  4705.     int min_s, max_s;
  4706.     hunits min_a, max_a;
  4707.     if (has_arg()
  4708.     && get_number(&min_s, 'z')
  4709.     && get_hunits(&min_a, 'p')
  4710.     && get_number(&max_s, 'z')
  4711.     && get_hunits(&max_a, 'p')) {
  4712.       track_kerning_function tk(min_s, min_a, max_s, max_a);
  4713.       font_table[n]->set_track_kern(tk);
  4714.     }
  4715.     else {
  4716.       track_kerning_function tk;
  4717.       font_table[n]->set_track_kern(tk);
  4718.     }
  4719.   }
  4720.   skip_line();
  4721. }
  4722.  
  4723. void constant_space()
  4724. {
  4725.   int n = get_fontno();
  4726.   if (n >= 0) {
  4727.     int x, y;
  4728.     if (!has_arg() || !get_integer(&x))
  4729.       font_table[n]->set_constant_space(CONSTANT_SPACE_NONE);
  4730.     else {
  4731.       if (!has_arg() || !get_number(&y, 'z'))
  4732.     font_table[n]->set_constant_space(CONSTANT_SPACE_RELATIVE, x);
  4733.       else
  4734.     font_table[n]->set_constant_space(CONSTANT_SPACE_ABSOLUTE, 
  4735.                       scale(y*x,
  4736.                         units_per_inch,
  4737.                         36*72*sizescale));
  4738.     }
  4739.   }
  4740.   skip_line();
  4741. }
  4742.  
  4743. void ligature()
  4744. {
  4745.   int lig;
  4746.   if (has_arg() && get_integer(&lig) && lig >= 0 && lig <= 2)
  4747.     global_ligature_mode = lig;
  4748.   else
  4749.     global_ligature_mode = 1;
  4750.   skip_line();
  4751. }
  4752.  
  4753. void kern_request()
  4754. {
  4755.   int k;
  4756.   if (has_arg() && get_integer(&k))
  4757.     global_kern_mode = k != 0;
  4758.   else
  4759.     global_kern_mode = 1;
  4760.   skip_line();
  4761. }
  4762.  
  4763. void set_soft_hyphen_char()
  4764. {
  4765.   soft_hyphen_char = get_optional_char();
  4766.   if (!soft_hyphen_char)
  4767.     soft_hyphen_char = get_charinfo(HYPHEN_SYMBOL);
  4768.   skip_line();
  4769. }
  4770.  
  4771. void init_output()
  4772. {
  4773.   if (suppress_output_flag)
  4774.     the_output = new suppress_output_file;
  4775.   else if (ascii_output_flag)
  4776.     the_output = new ascii_output_file;
  4777.   else
  4778.     the_output = new troff_output_file;
  4779. }
  4780.  
  4781. class next_available_font_position_reg : public reg {
  4782. public:
  4783.   const char *get_string();
  4784. };
  4785.  
  4786. const char *next_available_font_position_reg::get_string()
  4787. {
  4788.   return itoa(next_available_font_position());
  4789. }
  4790.  
  4791. class printing_reg : public reg {
  4792. public:
  4793.   const char *get_string();
  4794. };
  4795.  
  4796. const char *printing_reg::get_string()
  4797. {
  4798.   if (the_output)
  4799.     return the_output->is_printing() ? "1" : "0";
  4800.   else
  4801.     return "0";
  4802. }
  4803.  
  4804. void init_node_requests()
  4805. {
  4806.   init_request("fp", font_position);
  4807.   init_request("sty", style);
  4808.   init_request("cs", constant_space);
  4809.   init_request("bd", bold_font);
  4810.   init_request("uf", underline_font);
  4811.   init_request("lg", ligature);
  4812.   init_request("kern", kern_request);
  4813.   init_request("tkf", track_kern);
  4814.   init_request("special", special_request);
  4815.   init_request("fspecial", font_special_request);
  4816.   init_request("ftr", font_translate);
  4817.   init_request("shc", set_soft_hyphen_char);
  4818.   number_reg_dictionary.define(".fp", new next_available_font_position_reg);
  4819.   number_reg_dictionary.define(".kern",
  4820.                    new constant_int_reg(&global_kern_mode));
  4821.   number_reg_dictionary.define(".lg",
  4822.                    new constant_int_reg(&global_ligature_mode));
  4823.   number_reg_dictionary.define(".P", new printing_reg);
  4824.   soft_hyphen_char = get_charinfo(HYPHEN_SYMBOL);
  4825. }
  4826.